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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13330정성태4/26/20233525Windows: 247. Win32 C/C++ - CS_GLOBALCLASS 설명
13329정성태4/24/20233720Windows: 246. Win32 C/C++ - 직접 띄운 대화창 템플릿을 위한 Modal 메시지 루프 생성파일 다운로드1
13328정성태4/19/20233394VS.NET IDE: 184. Visual Studio - Fine Code Coverage에서 동작하지 않는 Fake/Shim 테스트
13327정성태4/19/20233803VS.NET IDE: 183. C# - .NET Core/5+ 환경에서 Fakes를 이용한 단위 테스트 방법
13326정성태4/18/20235228.NET Framework: 2109. C# - 닷넷 응용 프로그램에서 SQLite 사용 (System.Data.SQLite) [1]파일 다운로드1
13325정성태4/18/20234521스크립트: 48. 파이썬 - PostgreSQL의 with 문을 사용한 경우 연결 개체 누수
13324정성태4/17/20234353.NET Framework: 2108. C# - Octave의 "save -binary ..."로 생성한 바이너리 파일 분석파일 다운로드1
13323정성태4/16/20234281개발 환경 구성: 677. Octave에서 Excel read/write를 위한 io 패키지 설치
13322정성태4/15/20235059VS.NET IDE: 182. Visual Studio - 32비트로만 빌드된 ActiveX와 작업해야 한다면?
13321정성태4/14/20233886개발 환경 구성: 676. WSL/Linux Octave - Python 스크립트 연동
13320정성태4/13/20233863개발 환경 구성: 675. Windows Octave 8.1.0 - Python 스크립트 연동
13319정성태4/12/20234304개발 환경 구성: 674. WSL 2 환경에서 GNU Octave 설치
13318정성태4/11/20234138개발 환경 구성: 673. JetBrains IDE에서 "Squash Commits..." 메뉴가 비활성화된 경우
13317정성태4/11/20234232오류 유형: 855. WSL 2 Ubuntu 20.04 - error: cannot communicate with server: Post http://localhost/v2/snaps/...
13316정성태4/10/20233560오류 유형: 854. docker-compose 시 "json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)" 오류 발생
13315정성태4/10/20233755Windows: 245. Win32 - 시간 만료를 갖는 컨텍스트 메뉴와 윈도우 메시지의 영역별 정의파일 다운로드1
13314정성태4/9/20233844개발 환경 구성: 672. DosBox를 이용한 Turbo C, Windows 3.1 설치
13313정성태4/9/20233931개발 환경 구성: 671. Hyper-V VM에 Turbo C 2.0 설치 [2]
13312정성태4/8/20233943Windows: 244. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (개선된 버전)파일 다운로드1
13311정성태4/7/20234456C/C++: 163. Visual Studio 2022 - DirectShow 예제 컴파일(WAV Dest)
13310정성태4/6/20234039C/C++: 162. Visual Studio - /NODEFAULTLIB 옵션 설정 후 수동으로 추가해야 할 library
13309정성태4/5/20234215.NET Framework: 2107. .NET 6+ FileStream의 구조 변화
13308정성태4/4/20234111스크립트: 47. 파이썬의 time.time() 실숫값을 GoLang / C#에서 사용하는 방법
13307정성태4/4/20233878.NET Framework: 2106. C# - .NET Core/5+ 환경의 Windows Forms 응용 프로그램에서 HINSTANCE 구하는 방법
13306정성태4/3/20233672Windows: 243. Win32 - 윈도우(cbWndExtra) 및 윈도우 클래스(cbClsExtra) 저장소 사용 방법
13305정성태4/1/20234040Windows: 242. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (쉬운 버전)파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...