Microsoft MVP성태의 닷넷 이야기
글쓴 사람
홈페이지
첨부 파일

ServicePointManager.DefaultConnectionLimit와 HttpClient의 관계

이전 글에서,

System.Net.ServicePointManager의 DefaultConnectionLimit 속성 설명
; https://www.sysnet.pe.kr/2/0/10927

ServicePointManager.DefaultConnectionLimit의 역할
; https://www.sysnet.pe.kr/2/0/10929

ServicePointManager.DefaultConnectionLimit 값이 마이크로소프트가 제공하는 HttpWebRequest(및 그것을 내부적으로 사용하는 WebClient) 객체 등에 일종의 풀링 효과를 갖는다고 설명했습니다. 이로 인해 ASP.NET 환경의 경우 CLR 버전에 따라 DefaultConnectionLimit 기본값 설정이 다음과 같이 되어 있는데,

[ASP.NET + CLR 2.0]
    CPU(코어) 수 * 12

[ASP.NET + CLR 4.0]
    2147483647

.NET Core 환경에서는 다시 그 값이 무조건 2로 바뀌었습니다. 아니... 왜 이렇게 작은 것일까요?




이것의 배경에는 새로 나온 HttpClient의 영향이 있는 듯합니다.

HttpClient Class
; https://docs.microsoft.com/en-us/dotnet/api/system.net.http.httpclient?view=netframework-4.8

개인적으로 이 클래스의 이름은 다소 오해의 소지가 있다고 보입니다. 왜냐하면 그것의 단일 인스턴스 하나가 내부적으로는 요청을 발생시키는 메서드 호출에 따라 각각의 소켓을 생성하기 때문인데, 따라서 미적 요인 없이 이름을 짓는다면 HttpClient_CreateSocketPerCall 정도로 이해하시면 됩니다.

덕분에 닷넷 응용 프로그램에서는 전역적으로 "static" HttpClient 인스턴스를 하나만 가지고 있어도 됩니다.

static HttpClient _httpClient = new HttpClient();

당연히 HttpClient의 멤버는 thread-safe하기 때문에 다음과 같이 다중 스레드에서도 안전하게 요청을 발생시킬 수 있습니다.

// 실행 환경: .NET Framework 4.8

static void Main(string[] args)
{
    for (int i = 0; i < 3; i++)
    {
        ThreadPool.QueueUserWorkItem(requestHttp, null);
    }
}

private static void requestHttp(object state)
{
    requestHttpAsync(state).GetAwaiter().GetResult();
}

private static async Task requestHttpAsync(object state)
{
    try
    {
        string result = await _httpClient.GetStringAsync("http://test.mypc.com:8035");
        Console.WriteLine(result);
    }
    catch { }
}

실제로 위의 코드를 실행하고 대상 웹 서버에서 요청 처리를 지연시키도록 만들어 그 사이 netstat로 소켓을 확인해 보면,

C:\Windows\System32> netstat -ano | findstr 8035
  TCP    127.0.0.1:2927         127.0.0.1:8035         ESTABLISHED     7764
  TCP    127.0.0.1:2928         127.0.0.1:8035         ESTABLISHED     7764
  TCP    127.0.0.1:2929         127.0.0.1:8035         ESTABLISHED     7764
...[생략]...

저렇게 3개의 Socket이 생성된 것을 볼 수 있습니다.




그렇다면 단일 HttpClient 인스턴스가 관리하는 동시 연결 제어는 어떻게 하는 걸까요? 아쉽게도 이것이 .NET Framework과 .NET Core 환경에 따라 다릅니다. 우선, .NET Framework는 기존의 ServicePointManager.DefaultConnectionLimit의 영향을 받도록 HttpClient가 만들어져 있습니다.

실제로 지난 글에서 예제 코드로 작성한 service_point_sample 프로젝트를 다음과 같이 HttpClient로 바꿔 테스트를 해보면,

static HttpClient _httpClient = new HttpClient();

static void Main(string[] args)
{
    // ...[생략]...

    _httpClient.DefaultRequestHeaders.Add("Delay", "5000");

    // ...[생략]...

    for (int i = 0; i < reqCount; i ++)
    {
        ThreadPool.QueueUserWorkItem(requestHttp, urls[i % urls.Length]);
    }

    // ...[생략]...
}

private static void requestHttp(object state)
{
    requestHttpAsync(state).GetAwaiter().GetResult();
}

private static async Task requestHttpAsync(object state)
{
    // ...[생략]...

    try
    {
        string result = await _httpClient.GetStringAsync(url);
    }
    catch { }
}

해당 프로젝트가 .NET Framework의 Console 환경이었기 때문에 ServicePointManager.DefaultConnectionLimit == 2이므로, HttpWebRequest를 사용했을 때와 동일하게 다음과 같이 3번째 요청이 5초 후에 처리되는 것을 확인할 수 있습니다.

[1]2019-09-17 오후 1:23:29
GET / HTTP/1.1
Delay: 5000
Host: test.mypc.com:8035
Connection: Keep-Alive

[2]2019-09-17 오후 1:23:29
GET / HTTP/1.1
Delay: 5000
Host: test.mypc.com:8035
Connection: Keep-Alive

[3]2019-09-17 오후 1:23:34
GET / HTTP/1.1
Delay: 5000
Host: test.mypc.com:8035

반면, HttpClient를 전역 static으로 사용하지 않고 개별적으로 생성해 사용하면,

private static async Task requestHttpAsync(object state)
{
    // ...[생략]...

    using (HttpClient httpClient = new HttpClient())
    {
        try
        {
            string result = await httpClient.GetStringAsync(url);
            Console.WriteLine("Ended");
        }
        catch { }
    }
}

(DefaultConnectionLimit 값에 따라 개별 인스턴스마다 2개의 동시 요청을 생성할 수 있기 때문에, 사실상 제약 없이) 다음과 같이 3개의 GET 통신이 동시에 발생한 것을 확인할 수 있습니다.

[2]2019-09-17 오후 1:27:33
GET / HTTP/1.1
Host: test.mypc.com:8035
Connection: Keep-Alive


[1]2019-09-17 오후 1:27:33
GET / HTTP/1.1
Host: test.mypc.com:8035
Connection: Keep-Alive


[3]2019-09-17 오후 1:27:33
GET / HTTP/1.1
Host: test.mypc.com:8035
Connection: Keep-Alive




동일한 프로그램을 .NET Core 프로젝트로 테스트해보면 어떤 결과가 나올까요?

static HttpClient 단일 인스턴스를 생성해 테스트해도 이번에는 무조건 3개의 GET 통신이 동시에 발생하는 것을 볼 수 있습니다. 왜냐하면 .NET Core의 HttpClient는 ServicePointManager.DefaultConnectionLimit의 영향을 받지 않기 때문입니다. 아마도, HttpClient는 .NET Core 1.0부터 지원했던 반면 ServicePointManager는 2.0부터 지원이 되었기 때문에 독립적으로 동작을 하게 된 것이 아닌가 짐작이 됩니다.

따라서, .NET Core의 HttpClient는 다른 방식으로 동시 연결 제어를 다뤄야 했는데, 그것이 바로 HttpClient의 생성자에 HttpClientHandler를 이용한 MaxConnectionsPerServer 속성입니다.

static HttpClient _httpClient = new HttpClient(new HttpClientHandler { MaxConnectionsPerServer = 2 });
// HttpClientHandler에 MaxConnectionsPerServer를 지정하지 않은 경우 기본값은 2147483647

위와 같이 설정한 _httpClient 인스턴스는 이제 .NET Framework 환경에서 ServicePointManager.DefaultConnectionLimit == 2를 지정했을 때와 동일하게 2개의 동시 연결만을 지원하도록 동작합니다.

그건 그렇고, 개인적으로 .NET Core 2.0에 구현된 ServicePointManager.DefaultConnectionLimit의 역할을 잘 모르겠습니다. (혹시 아시는 분은 덧글 부탁드립니다. ^^) 왜냐하면 .NET Core 환경에서는 HttpWebRequest(또는 그것을 내부적으로 사용하는 WebClient)가 내부적으로는 HttpClient를 생성하기 때문에 ServicePointManager.DefaultConnectionLimit와 무관하게 동작하며, 심지어 내부의 MaxConnectionsPerServer 값을 제어할 방법도 없기 때문에 무조건 동시 연결 제어 없이 동작합니다.

더군다나, 조심스럽긴 하지만 테스트해본 결과에 의하면 .NET Core의 HttpWebRequest는 HttpClient의 다중 소켓 지원을 받으므로 마찬가지로 단일 static 인스턴스로 다중 스레드에서 사용해도 오류 없이 잘 동작합니다.




HttpClient의 이렇게 복잡한 소켓 관리 기능으로 인해 전역적으로 사용하지 않고 개별 건마다 사용하는 경우에는 반드시 Dispose 메서드를 호출하는 것이 좋습니다.

using (HttpClient httpClient = new HttpClient())
{
    string result = await httpClient.GetStringAsync(url);
}

그렇지 않으면 내부에 생성한 자원들이 다음 번 GC되는 순간까지 지연되어 해제되기 때문에 경우에 따라 응용 프로그램에 심각한 성능 문제를 야기할 수 있습니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]





[최초 등록일: ]
[최종 수정일: 9/19/2019 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer@outlook.com

비밀번호

댓글 쓴 사람
 




[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
12041정성태10/23/20193오류 유형: 572. mstest.exe - The load test results database could not be opened.
12040정성태10/23/20194오류 유형: 571. Unhandled Exception: System.Net.Mail.SmtpException: Transaction failed. The server response was: 5.2.0 STOREDRV.Submission.Exception:SendAsDeniedException.MapiExceptionSendAsDenied
12039정성태10/22/201915스크립트: 16. cmd.exe의 for 문에서는 ERRORLEVEL이 설정되지 않는 문제
12038정성태10/17/201932오류 유형: 570. SQL Server 2019 RC1 - SQL Client Connectivity SDK 설치 오류
12037정성태10/15/2019132.NET Framework: 867. C# - Encoding.Default 값을 바꿀 수 있을까요?파일 다운로드1
12036정성태10/21/201998.NET Framework: 866. C# - 고성능이 필요한 환경에서 GC가 발생하지 않는 네이티브 힙 사용파일 다운로드1
12035정성태10/13/201997개발 환경 구성: 461. C# 8.0의 #nulable 관련 특성을 .NET Framework 프로젝트에서 사용하는 방법파일 다운로드1
12034정성태10/12/2019157개발 환경 구성: 460. .NET Core 환경에서 (프로젝트가 아닌) C# 코드 파일을 입력으로 컴파일하는 방법 [1]
12033정성태10/11/2019114개발 환경 구성: 459. .NET Framework 프로젝트에서 C# 8.0 컴파일러를 사용하는 방법
12032정성태10/8/201990.NET Framework: 865. .NET Core 2.2/3.0 웹 프로젝트를 IIS에서 호스팅(Inproc, out-of-proc)하는 방법 - AspNetCoreModuleV2 소개
12031정성태10/7/201932오류 유형: 569. Azure Site Extension 업그레이드 시 "System.IO.IOException: There is not enough space on the disk" 예외 발생
12030정성태10/21/2019892.NET Framework: 864. .NET Conf 2019 Korea - "닷넷 17년의 변화 정리 및 닷넷 코어 3.0" 발표 자료 [1]파일 다운로드1
12029정성태9/27/2019108제니퍼 .NET: 29. Jennifersoft provides a trial promotion on its APM solution such as JENNIFER, PHP, and .NET in 2019 and shares the examples of their application.
12028정성태9/26/2019135.NET Framework: 863. C# - Thread.Suspend 호출 시 응용 프로그램 hang 현상을 해결하기 위한 시도파일 다운로드1
12027정성태9/26/201949오류 유형: 568. Consider app.config remapping of assembly "..." from Version "..." [...] to Version "..." [...] to solve conflict and get rid of warning.
12026정성태9/26/201964.NET Framework: 862. C# - Active Directory의 LDAP 경로 및 정보 조회
12025정성태9/25/2019164제니퍼 .NET: 28. APM 솔루션 제니퍼, PHP, .NET 무료 사용 프로모션 2019 및 적용 사례 (8)
12024정성태9/20/2019152.NET Framework: 861. HttpClient와 HttpClientHandler의 관계
12023정성태9/19/2019192.NET Framework: 860. ServicePointManager.DefaultConnectionLimit와 HttpClient의 관계파일 다운로드1
12022정성태9/12/2019408개발 환경 구성: 458. C# 8.0 (Preview) 신규 문법을 위한 개발 환경 구성 [3]
12021정성태9/12/2019175[예약]
12020정성태9/11/2019135VC++: 134. SYSTEMTIME 값 기준으로 특정 시간이 지났는지를 판단하는 함수
12019정성태9/11/2019135Linux: 23. .NET Core + 리눅스 환경에서 Environment.CurrentDirectory 접근 시 주의 사항
12018정성태9/25/201984오류 유형: 567. IIS - Unrecognized attribute 'targetFramework'. Note that attribute names are case-sensitive. (D:\lowSite4\web.config line 11)
12017정성태9/11/2019303오류 유형: 566. 비주얼 스튜디오 - Failed to register URL "http://localhost:6879/" for site "..." application "/". Error description: Access is denied. (0x80070005)
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...