성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] 그냥 RSS Reader 기능과 약간의 UI 편의성 때문에 사용...
[이종효] 오래된 소프트웨어는 보안 위협이 되기도 합니다. 혹시 어떤 기능...
[정성태] @Keystroke IEEE의 문서를 소개해 주시다니... +_...
[손민수 (Keystroke)] 괜히 듀얼채널 구성할 때 한번에 같은 제품 사라고 하는 것이 아...
[정성태] 전각(Full-width)/반각(Half-width) 기능을 토...
[정성태] Vector에 대한 내용은 없습니다. Vector가 닷넷 BCL...
[orion] 글 읽고 찾아보니 디자인 타임에는 InitializeCompon...
[orion] 연휴 전에 재현 프로젝트 올리자 생각해 놓고 여의치 않아서 못 ...
[정성태] 아래의 글에 정리했으니 참고하세요. C# - Typed D...
[정성태] 간단한 재현 프로젝트라도 있을까요? 저런 식으로 설명만 해...
글쓰기
제목
이름
암호
전자우편
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# - 문자열 연결 시 string.Create를 이용한 GC 할당 최소화</h1> <p> 이번 글은 아래의 트윗 내용을 옮겨봅니다. ^^<br /> <br /> <blockquote class="twitter-tweet"><p lang="en" dir="ltr">String.Create is 🔥 for string concatenation ⬇ <br><br>Create & populate strings in one pass, avoid unnecessary memory copies, and skip intermediate allocations. <br><br>Ideal when the final string length is known and in high performance hot paths. <br><br>Have you used it?<a href="https://twitter.com/hashtag/dotnet?src=hash&ref_src=twsrc%5Etfw">#dotnet</a> <a href="https://twitter.com/hashtag/csharp?src=hash&ref_src=twsrc%5Etfw">#csharp</a> <a href="https://t.co/vzSMHXgunx">pic.twitter.com/vzSMHXgunx</a></p>— Dave Callan | dotnet (@Dave_DotNet) <a href="https://twitter.com/Dave_DotNet/status/1682699744747958274?ref_src=twsrc%5Etfw">July 22, 2023</a></blockquote> <script async src="https://platform.twitter.com/widgets.js" charset="utf-8"></script><br /> <br /> 예전에도 "<a target='tab' href='https://www.sysnet.pe.kr/2/0/12826'>C# 10 - (12) 문자열 보간 성능 개선</a>" 글에서 string.Create를 스치듯 다룬 적이 있었습니다. ^^<br /> <br /> 어쨌든 중요한 것은, string 자체는 참조 타입이라서 GC Heap을 쓸 수밖에 없다는 점입니다. 하지만, string을 연결하는 과정에서 가능한 stack을 활용해 GC 힙의 사용을 최소화하는 노력은 할 수 있습니다.<br /> <br /> 위의 트윗에서 나온 코드를 실습해 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > namespace ConsoleApp1; internal class Program { static void Main(string[] args) { { string text = StringCreate(); // JIT Console.WriteLine(text.Length * 2); } { long old = GC.GetAllocatedBytesForCurrentThread(); string text = StringCreate(); Console.WriteLine(text); long now = GC.GetAllocatedBytesForCurrentThread(); Console.WriteLine(now - old); } } static private string title = "Mr."; static private string first = "David"; static private string middle = "Patrick"; static private string last = "Callan"; static public string StringCreate() { string text = string.Create(<span style='color: blue; font-weight: bold'>title.Length + first.Length + middle.Length + last.Length + 3</span>, (title, first, middle, last), (span, state) => { state.title.AsSpan().CopyTo(span); span = span[state.title.Length..]; span[0] = ' '; span = span[1..]; state.first.AsSpan().CopyTo(span); span = span[state.first.Length..]; span[0] = ' '; span = span[1..]; state.middle.AsSpan().CopyTo(span); span = span[state.middle.Length..]; span[0] = ' '; span = span[1..]; state.last.AsSpan().CopyTo(span); } ); return text; } } </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;' > 48 Mr. David Patrick Callan 72 </pre> <br /> StringCreate를 실행했을 때 GC Heap을 72바이트 소비하는 것으로, 문자열 길이가 48바이트이므로 null 2바이트를 포함하면 50바이트, 그래도 22바이트가 더 소비되긴 했습니다. 어떻게 소비된 것인지 다음의 글에 따라 계산해 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > windbg - .NET string의 x86/x64 메모리 할당 구조 ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/11336'>https://www.sysnet.pe.kr/2/0/11336</a> </pre> <br /> <ul> <li>Object Header: 8바이트</li> <li>MethodTable 주소: 8바이트</li> <li>m_stringLength: 4바이트</li> <li>...[문자열 48바이트]...</li> <li>null 2바이트</li> <li>8바이트 정렬로 인해 2바이트</li> </ul> <br /> 모두 더해 정확히 72바이트입니다. ^^ 그러니까 결국 Span을 이용한 string.Create의 사용은 대상 문자열로 인한 GC 힙의 사용 외에는 나머지 할당을 완전히 없앤 것입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> ^^ 눈치채신 분이 있겠지만, 사실 위와 같이 코딩하는 것은 아래와 같이 바꿔쓸 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // C# 10+, .NET 6+ { <span style='color: blue; font-weight: bold'>string text = $"{title} {first} {middle} {last}";</span> } { long old = GC.GetAllocatedBytesForCurrentThread(); string text = $"{title} {first} {middle} {last}"; long now = GC.GetAllocatedBytesForCurrentThread(); Console.WriteLine(now - old); // 출력 결과: 72 } </pre> <br /> 위의 코드 역시 72바이트만을 소비하는데, "<a target='tab' href='https://www.sysnet.pe.kr/2/0/12826'>C# 10 - (12) 문자열 보간 성능 개선</a>"에서 설명한 대로 이미 <a target='tab' href='https://learn.microsoft.com/en-us/dotnet/api/system.runtime.compilerservices.defaultinterpolatedstringhandler'>DefaultInterpolatedStringHandler</a>가 내부적으로 string.Create를 이용한 문자열 연결을 하고 있기 때문입니다.<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
5219
(왼쪽의 숫자를 입력해야 합니다.)