Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
.NET Framework: 860. ServicePointManager.DefaultConnectionLimit와 HttpClient의 관계
; https://www.sysnet.pe.kr/2/0/12023

.NET Framework: 861. HttpClient와 HttpClientHandler의 관계
; https://www.sysnet.pe.kr/2/0/12024

닷넷: 2200.  C# - HttpClient.PostAsJsonAsync 호출 시 "Transfer-Encoding: chunked" 대신 "Content-Length" 헤더 처리
; https://www.sysnet.pe.kr/2/0/13522




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://learn.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되는 순간까지 지연되어 해제되기 때문에 경우에 따라 응용 프로그램에 심각한 성능 문제를 야기할 수 있습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/25/2023]

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

비밀번호

댓글 작성자
 




... 16  17  18  19  20  21  [22]  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13070정성태6/8/20227043오류 유형: 813. Windows 11에서 입력 포커스가 바뀌는 문제 [1]
13069정성태5/26/20229275.NET Framework: 2019. C# - .NET에서 제공하는 3가지 Timer 비교 [2]
13068정성태5/24/20227731.NET Framework: 2018. C# - 일정 크기를 할당하는 동안 GC를 (가능한) 멈추는 방법 [1]파일 다운로드1
13067정성태5/23/20227092Windows: 206. Outlook - 1년 이상 지난 메일이 기본적으로 안 보이는 문제
13066정성태5/23/20226405Windows: 205. Windows 11 - Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
13065정성태5/20/20227083.NET Framework: 2017. C# - Windows I/O Ring 소개 [2]파일 다운로드1
13064정성태5/18/20226643.NET Framework: 2016. C# - JIT 컴파일러의 인라인 메서드 처리 유무
13063정성태5/18/20227063.NET Framework: 2015. C# - 인라인 메서드(inline methods)
13062정성태5/17/20227851.NET Framework: 2014. C# - async/await 그리고 스레드 (4) 비동기 I/O 재현파일 다운로드1
13061정성태5/16/20226666.NET Framework: 2013. C# - FILE_FLAG_OVERLAPPED가 적용된 파일의 읽기/쓰기 시 Position 관리파일 다운로드1
13060정성태5/15/20229107.NET Framework: 2012. C# - async/await 그리고 스레드 (3) Task.Delay 재현파일 다운로드1
13059정성태5/14/20227606.NET Framework: 2011. C# - CLR ThreadPool의 I/O 스레드에 작업을 맡기는 방법 [1]파일 다운로드1
13058정성태5/13/20227529.NET Framework: 2010. C# - ThreadPool.SetMaxThreads 사용법
13057정성태5/12/20229199오류 유형: 812. 파이썬 - ImportError: cannot import name ...
13056정성태5/12/20226372.NET Framework: 2009. C# - async/await 그리고 스레드 (2) MyTask의 호출 흐름 [2]파일 다운로드1
13055정성태5/11/20229262.NET Framework: 2008. C# - async/await 그리고 스레드 (1) MyTask로 재현 [11]파일 다운로드1
13054정성태5/11/20226786.NET Framework: 2007. C# - 10진수 숫자를 담은 문자열을 숫자로 변환하는 방법 [11]파일 다운로드1
13053정성태5/10/20226417.NET Framework: 2006. C# - GC.KeepAlive 메서드의 역할
13052정성태5/9/20226425.NET Framework: 2005. C# - 생성한 참조 개체가 언제 GC의 정리 대상이 될까요?
13051정성태5/8/20226372.NET Framework: 2004. C# XingAPI - ACF 검색 결과로 구한 CSV 파일을 통해 퀀트 종목 찾기파일 다운로드1
13050정성태5/6/20226395.NET Framework: 2003. C# - COM 개체의 이벤트 핸들러에서 발생하는 예외에 대한 CLR의 특별 대우파일 다운로드1
13049정성태5/6/20225374오류 유형: 811. GoLand - Error: Cannot find package
13048정성태5/6/20226497오류 유형: 810. "ASUS TUF GAMING B550M-PLUS (WI-FI)" 모델에서 블루투스 장치가 인식이 안 되는 문제
13047정성태5/6/20226483오류 유형: 809. Speech Recognition could not start
13046정성태5/5/20226772.NET Framework: 2002. C# XingAPI - ACF 파일을 이용한 퀀트 종목 찾기(t1857)
13045정성태5/5/20226833.NET Framework: 2001. C# XingAPI - 주식 종목에 따른 PBR, PER, ROE 구하는 방법(t3341 예제)
... 16  17  18  19  20  21  [22]  23  24  25  26  27  28  29  30  ...