성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>C# - .NET Platform Extension의 ObjectPool<T> 사용법 소개</h1> <p> 예전에, ArrayPool과 MemoryPool에 대해 설명한 적이 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C# - ArrayPool<T>와 MemoryPool<T> 소개 ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12480'>https://www.sysnet.pe.kr/2/0/12480</a> </pre> <br /> 그렇다면 당연히 일반적인 개체에 대한 Pool 구현을 찾을 수밖에 없는데요, 재미있게도 기본적인 .NET Core/5+에 포함되지 않고 별도의 ".NET Platform Extension"이라는 구성 요소로 나와 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > ObjectPool<T> Class ; <a target='tab' href='https://learn.microsoft.com/en-us/dotnet/api/microsoft.extensions.objectpool.objectpool-1?view=dotnet-plat-ext-6.0'>https://learn.microsoft.com/en-us/dotnet/api/microsoft.extensions.objectpool.objectpool-1?view=dotnet-plat-ext-6.0</a> </pre> <br /> 저도 오늘 처음 저 문구가 눈에 들어왔는데요, ^^; 이에 대해 stackoverflow에 다음의 질문/답변이 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > What are .NET Platform Extensions on docs.microsoft.com? ; <a target='tab' href='https://stackoverflow.com/questions/53097067/what-are-net-platform-extensions-on-docs-microsoft-com'>https://stackoverflow.com/questions/53097067/what-are-net-platform-extensions-on-docs-microsoft-com</a> </pre> <br /> 그러니까, .NET Core/5+의 프레임워크에는 추가되지 못한 "확장 구성 요소"입니다. 이름에 "Platform"이 들어가 있어 플랫폼 의존적인 것이 있지 않을까 싶어도 실제 네임스페이스나 어셈블리 명을 보면 단순히 "Microsoft.Extensions."로 시작하므로 꼭 그렇다고 볼 수는 없을 듯합니다.<br /> <br /> 그나저나 도움말 문서를 보면,<br /> <br /> <img onclick='toggle_img(this)' class='imgView' alt='net_platform_ext_1.png' src='/SysWebRes/bbs/net_platform_ext_1.png' /><br /> <br /> 1.0부터 시작해 벌써 6.0까지 꾸준히 업데이트되었는데... 왜 저는 이제서야 그 이름을 들어본 걸까요? ^^<br /> <br /> <hr style='width: 50%' /><br /> <br /> ObjectPool에 대한 사용법은 아래의 글에서 이미 잘 소개하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Detailed Explanation of Object Pool’s Various Usages in.NET Core ; <a target='tab' href='https://developpaper.com/detailed-explanation-of-object-pools-various-usages-in-net-core/'>https://developpaper.com/detailed-explanation-of-object-pools-various-usages-in-net-core/</a> </pre> <br /> 저는 그냥 위의 글을 거의 베끼는 수준으로 정리만 해보겠습니다. ^^<br /> <br /> 자, 그럼 우선 NuGet으로부터 패키지 참조 추가를 해야겠지요?<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // 현재, 최신 버전의 Microsoft.Extensions.ObjectPool은 요구 사항이 .NET Standard 2.0이므로 .NET Framework 환경에서도 4.6.1 이상이면 사용이 가능합니다. // Install-Package Microsoft.Extensions.ObjectPool -Version 6.0.1 Install-Package Microsoft.Extensions.ObjectPool </pre> <br /> ObjectPool<T> 자체는 추상 타입입니다. 왜냐하면, 개체에 대한 생성자 호출이 제각각일 수 있기 때문입니다. 하지만, 마이크로소프트는 개체를 생성하는 작업을 제외하고는 모두 구현을 해두었고, 일반적인 상황에서는 new 작업만 해결하면 되므로 굳이 ObjectPool을 상속해 구현할 필요 없이 쉽게 사용할 수 있도록 DefaultObjectPool<T> 타입을 제공하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > DefaultObjectPool<T> Class ; <a target='tab' href='https://learn.microsoft.com/en-us/dotnet/api/microsoft.extensions.objectpool.defaultobjectpool-1?view=dotnet-plat-ext-6.0'>https://learn.microsoft.com/en-us/dotnet/api/microsoft.extensions.objectpool.defaultobjectpool-1?view=dotnet-plat-ext-6.0</a> </pre> <br /> 예를 들어, Socket 개체를 10개 풀링한다고 가정해 보겠습니다. 그럼 다음과 같이 DefaultObjectPool을 사용해 자신만의 ObjectPool을 만들 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > static void Main(string[] args) { IPooledObjectPolicy<Socket> policy = ...[생략]...; // 최댓수 10을 지정하지 않으면, 기본값은 Environment.ProcessorCount * 2 <span style='color: blue; font-weight: bold'>DefaultObjectPool<Socket> pool = new DefaultObjectPool<Socket>(policy, 10);</span> } </pre> <br /> 하지만, 우리가 원하는 것은 특정 서버로 연결된 소켓일 텐데요, 이를 위해 소켓 생성을 우리가 명시해야 하는데 이 과정을 IPooledObjectPolicy<T> 인터페이스만 구현한 코드로 해결할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public class PooledObjectPolicy : IPooledObjectPolicy<Socket> { string _host; int _port; public PooledObjectPolicy(string host, int port) { _host = host; _port = port; } public Socket <span style='color: blue; font-weight: bold'>Create</span>() { Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); socket.Connect(_host, _port); return socket; } public bool <span style='color: blue; font-weight: bold'>Return</span>(Socket obj) { return true; } } </pre> <br /> 바로 저 PooledObjectPolicy 클래스의 인스턴스를 ObjectPool의 인자에 전달하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <span style='color: blue; font-weight: bold'>IPooledObjectPolicy<Socket> policy = new PooledObjectPolicy("localhost", 39_999);</span> DefaultObjectPool<Socket> pool = new DefaultObjectPool<Socket>(<span style='color: blue; font-weight: bold'>policy</span>, 10); </pre> <br /> 개체 풀링이 이뤄지는 것입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > for (int i = 0; i < 3; i++) { Socket socket = pool.Get(); // 개체를 요청하고, pool.Return(socket); // 곧바로 반환하므로, 실제로는 1개의 Socket 개체만 생성됨 } </pre> <br /> <hr style='width: 50%' /><br /> <br /> "풀링"에 대한 기본 동작 방식은 <a target='tab' href='https://www.sysnet.pe.kr/2/0/12478'>ArrayPool</a>과 같습니다. 즉, "개체 풀링"이라고 해서 그동안 흔히 사용했던 ADO.NET의 연결 개체 풀링처럼 동작하는 것은 아니라는 점입니다.<br /> <br /> 가장 큰 차이점이라면, (위의 예제에서 최댓값 10을 지정했는데) 10개의 개체를 다 소진한 이후에도 여전히 (아무런 지연 없이) 새로운 개체를 Get 메서드를 이용해 요청할 수 있다는 점입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > for (int i = 0; i < <span style='color: blue; font-weight: bold'>11</span>; i++) { Socket socket = pool.Get(); // 11번의 Get 요청이지만 지연 없이 11개의 개체를 반환 } </pre> <br /> 단지, 10개까지는 Return을 통해 풀에 반환할 수 있는 반면 11개 이후의 개체는 반환이 되지 않고 GC 대상이 될 뿐입니다. 이에 대한 테스트를 다음과 같이 할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > List<Socket> sockets = new List<Socket>(); for (int i = 0; i < <span style='color: blue; font-weight: bold'>11</span>; i++) { sockets.Add(pool.Get()); // 이 루프에서 11개의 Socket 개체가 생성 } Console.WriteLine(); sockets.ForEach((elem) => pool.Return(elem)); // 11개의 개체를 반환한다고 호출해도 실제로는 10개만 반환 sockets.Clear(); for (int i = 0; i < <span style='color: blue; font-weight: bold'>11</span>; i++) { sockets.Add(pool.Get()); // 이 루프에서는 10개는 풀로부터 반환되고, 1개만 새롭게 생성해 반환 } </pre> <br /> <hr style='width: 50%' /><br /> <br /> 동작 방식이 저렇기 때문에 고성능을 요구하는 환경에서 GC의 수를 줄일 수 있으면서도, 설령 개체를 풀에 반환하는 동작에 버그가 있다고 해도 후속 요청에서 (DB 연결과는 달리) 지연 현상 없이 곧바로 개체를 사용할 수 있다는 장점이 있습니다.<br /> <br /> 근래의 C# 10과 각종 Pool 개체를 잘 조합하면 정말 GC가 거의 발생하지 않는 C# 프로그램도 꿈은 아닐 듯합니다. 아마도 위의 방식들을 조합해서 .NET 6의 ASP.NET 관련 성능을 끌어올렸을 것입니다.<br /> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1876&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1771
(왼쪽의 숫자를 입력해야 합니다.)