Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 4개 있습니다.)

닷넷 (및 VM 계열 언어) 코드의 성능 측정 시 주의할 점

아래와 같은 글이 있군요. ^^

c# 에서 문자열 계산속도
; http://www.gamecodi.com/board/zboard-id-GAMECODI_Talkdev-no-5247-z-2.htm

그런데, 위의 코드는 잘못된 방식의 성능 측정을 하고 있습니다. 왜냐하면 초기 JIT 컴파일을 생각하지 않고 작성한 코드인데 정상적으로 바꿔보면 다음과 같이 테스트할 수 있습니다.

using System;
using System.Diagnostics;

class Program
{
    static void Main(string[] args)
    {
        Action<int, string, Action<int, string>, string> action = (loopCount, title, work, arg) =>
        {
            Stopwatch st = new Stopwatch();
            st.Start();

            work(loopCount, arg);

            st.Stop();

            Console.WriteLine(title + " : " + st.ElapsedMilliseconds);
        };

        action(1, "ConcatText for JIT", ConcatText, "");
        action(1, "InterpolateText for JIT", InterpolateText, "");
        action(1, "FormatText for JIT", FormatText, "");

        Console.WriteLine();

        action(10000000, "ConcatText", ConcatText, "");
        action(10000000, "InterpolateText", InterpolateText, "");
        action(10000000, "FormatText", FormatText, "");
    }

    private static void ConcatText(int loopCount, string text)
    {
        for (int i = 0; i < loopCount; i ++)
        {
            TestLog(text + "a");
        }
    }

    private static void InterpolateText(int loopCount, string text)
    {
        for (int i = 0; i < loopCount; i++)
        {
            TestLog($"{text}a");
        }
    }

    private static void FormatText(int loopCount, string text)
    {
        for (int i = 0; i < loopCount; i++)
        {
            TestLog("{0}a", text);
        }
    }

    static void TestLog(string format, params object [] args)
    {
    }
}

보는 바와 같이 JIT 컴파일로 인한 부하를 제거하고 다시 본격적인 성능 측정을 하도록 바뀌었습니다. 그래서 "c# 에서 문자열 계산속도" 글에서의 출력과,

String Concat took: 150 ms
String Interpolation took: 141 ms
String Format took: 191 ms

제가 테스트한 출력 결과를 보면,

[x86 - Debug 빌드]

ConcatText : 116
InterpolateText : 116
FormatText : 120

결과가 다릅니다. 그런데 이것을 릴리스 빌드로 하면 다시 차이가 발생합니다.

[x86 - Release 빌드]

ConcatText : 100
InterpolateText : 33
FormatText : 70

재미있는 것은 JIT 컴파일 방식이 개선된 x64에서는 다시 이런 결과가 나옵니다.

[x64 - Debug 빌드]

ConcatText : 175
InterpolateText : 114
FormatText : 111

[x64 - Release 빌드]

ConcatText : 80
InterpolateText : 31
FormatText : 80

그런데, 아직 끝이 아닙니다. ^^




(이하, 테스트는 x64 - Release 빌드의 결과만 다룹니다.)

잠시 살펴보면, 마지막 Format 코드에 대한 테스트가 잘못되었습니다. 원래 의도는 string.Format을 호출하는 것이었을 텐데 단순히 TestLog에 인자를 넘기는 동작만 하기 때문에 사실 문자열 연산과는 크게 관련이 없습니다. 따라서 원래 의도를 반영하려면 다음과 같이 바꿔야 하고,

private static void FormatText(int loopCount, string text)
{
    for (int i = 0; i < loopCount; i++)
    {
        TestLog(string.Format("{0}a", text));
    }
}

이렇게 되면 출력 결과가 완전히 달라집니다.

[x64 - Release 빌드]

ConcatText : 77
InterpolateText : 31
FormatText : 716

여기서 더욱 재미있는 것은, IL 코드입니다. 다르게 작성한 ConcatText, InterpolateText 메서드는,

private static void ConcatText(int loopCount, string text)
{
    for (int i = 0; i < loopCount; i++)
    {
        TestLog(text + "a");
    }
}

private static void InterpolateText(int loopCount, string text)
{
    for (int i = 0; i < loopCount; i++)
    {
        TestLog($"{text}a");
    }
}

다음과 같이 동일한 IL 코드로 번역됩니다.

for (int i = 0; i < loopCount; i++)
{
    TestLog(text + "a", Array.Empty<object>());
}

그러니까, C# 컴파일러는 $"{text}"a 코드를 원래는 string.Format으로 바꿔야 하지만 어차피 2개의 문자열을 하나로 합치는 것이므로 단순히 concat 연산자로 처리하는 것이 더 빠르다고 판단해 최적화를 해버립니다.

아니... 근데, 그럼 IL 코드가 같은 그 2개의 메서드가 성능이 왜 다른 것일까요? 저도 모르겠습니다. ^^; 단지 현상을 보려면 JIT 컴파일된 기계어를 보면 됩니다. 다음은 ConcatText 메서드가 JIT 컴파일된 결과입니다.

// ConcatText

    33:         for (int i = 0; i < loopCount; i++)
00007FF9DD360D32 53                   push        rbx  
00007FF9DD360D33 48 83 EC 20          sub         rsp,20h  
00007FF9DD360D37 8B F1                mov         esi,ecx  
00007FF9DD360D39 48 8B FA             mov         rdi,rdx  
00007FF9DD360D3C 33 DB                xor         ebx,ebx  
00007FF9DD360D3E 85 F6                test        esi,esi  
00007FF9DD360D40 7E 2A                jle         00007FF9DD360D6C  
    34:         {
    35:             TestLog(text + "a");
00007FF9DD360D42 48 BA 90 36 00 90 76 01 00 00 mov         rdx,17690003690h  
00007FF9DD360D4C 48 8B 12             mov         rdx,qword ptr [rdx]  
00007FF9DD360D4F 48 8B CF             mov         rcx,rdi  
00007FF9DD360D52 E8 59 87 F9 5C       call        00007FFA3A2F94B0  
00007FF9DD360D57 B9 01 00 00 00       mov         ecx,1  
00007FF9DD360D5C BA 53 00 00 00       mov         edx,53h  
00007FF9DD360D61 E8 6A 22 6E 5F       call        00007FFA3CA42FD0  
    33:         for (int i = 0; i < loopCount; i++)
00007FF9DD360D66 FF C3                inc         ebx  
00007FF9DD360D68 3B DE                cmp         ebx,esi  
00007FF9DD360D6A 7C D6                jl          00007FF9DD360D42  
00007FF9DD360D6C 48 83 C4 20          add         rsp,20h  
00007FF9DD360D70 5B                   pop         rbx  
00007FF9DD360D71 5E                   pop         rsi  
00007FF9DD360D72 5F                   pop         rdi  
00007FF9DD360D73 C3                   ret  
}

반면 InterpolateText는 (ConcatText와 동일한 IL 코드였음에도 불구하고) 다음과 같이 call 하나가 빠져 있습니다.

// InterpolateText

    41:         for (int i = 0; i < loopCount; i++)
00007FF9DD380D92 53                   push        rbx  
00007FF9DD380D93 48 83 EC 20          sub         rsp,20h  
00007FF9DD380D97 8B F1                mov         esi,ecx  
00007FF9DD380D99 48 8B FA             mov         rdi,rdx  
00007FF9DD380D9C 33 DB                xor         ebx,ebx  
00007FF9DD380D9E 85 F6                test        esi,esi  
00007FF9DD380DA0 7E 1B                jle         00007FF9DD380DBD  
    42:         {
    43:             TestLog($"{text}a");
00007FF9DD380DA2 48 BA 90 36 BC 64 6E 01 00 00 mov         rdx,16E64BC3690h  
00007FF9DD380DAC 48 8B 12             mov         rdx,qword ptr [rdx]  
00007FF9DD380DAF 48 8B CF             mov         rcx,rdi  
00007FF9DD380DB2 E8 F9 86 F7 5C       call        00007FFA3A2F94B0  
    41:         for (int i = 0; i < loopCount; i++)
00007FF9DD380DB7 FF C3                inc         ebx  
00007FF9DD380DB9 3B DE                cmp         ebx,esi  
00007FF9DD380DBB 7C E5                jl          00007FF9DD380DA2  
00007FF9DD380DBD 48 83 C4 20          add         rsp,20h  
00007FF9DD380DC1 5B                   pop         rbx  
00007FF9DD380DC2 5E                   pop         rsi  
00007FF9DD380DC3 5F                   pop         rdi  
00007FF9DD380DC4 C3                   ret  
}

이유는 알 수 없지만, 어쨌든 x64 JIT 컴파일러는 한 쪽 메서드에만 다음의 call 하나를 추가한 탓에 성능 차이가 발생합니다.

00007FF9DD360D57 B9 01 00 00 00       mov         ecx,1  
00007FF9DD360D5C BA 53 00 00 00       mov         edx,53h  
00007FF9DD360D61 E8 6A 22 6E 5F       call        00007FFA3CA42FD0  

첫 번째 인자에 1을 넣고 두 번째 인자에 83 (0x53)을 넣어 함수를 호출하는데 도대체 알 수 없는 코드입니다. ^^;

정리해 보면, (위의 환경에서) ConcatText나 InterpolateText는 완전히 같은 처리를 합니다. 따라서 string.Format으로 명시적인 호출을 발생하는 FormatText만 유독 크게 성능이 떨어지는 것처럼 보입니다.

반면, Interpolate의 문자열이 좀 더 복잡해진다면... 그래서2 C# 컴파일러가 string.Format으로 번역하게 된다면 이번에는 InterpolateText와 FormatText의 성능이 같을 것입니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




그러니까... C#과 같은 VM 환경의 언어에서는 성능 측정이 매우 까다롭습니다. 그래서, 어느 게 좋고 어느 게 나쁘다는 것을 판정하려면 꽤나 주의를 기울여야 하고 따질 것이 많습니다. 경험상, 닷넷 코드의 성능을 측정 시에는 적어도 다음의 요인들을 잘 살펴야 합니다.

  • 플랫폼 및 빌드 모드 표시(경우에 따라, .NET Framework 버전 명시)
  • [실행 시점] 초기 JIT 컴파일 시간 영향 제거
  • [컴파일 시점] C# 컴파일 단계에서의 최적화(Debug/Release)로 인한 차이
  • [실행 시점] JIT 컴파일 단계에서의 최적화(Debug/Release)로 인한 차이
  • [실행 시점] JIT 컴파일 단계에서 Debugger의 attach 상태 유무에 따른 최적화로 인한 차이




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/18/2024]

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

비밀번호

댓글 작성자
 



2019-08-14 01시55분
[폴리모프] (^-^)b
[guest]
2019-08-20 01시05분
저도 겜코디 쓰레드 읽으면서 뭔가 이상한데라고 느꼈는데, 이렇게 테스트하셔서 글을 올려주셨네요! 감사합니다.

성능 측정하는 글들이 여러군데 흩어져 있는데, 이번에 나온 책에서는 저 부분이 어느정도 모아져 있어 참고하기 괜찮았었네요!

- Pro .NET Benchmarking : https://www.apress.com/gp/book/9781484249406
Eunpyoung Kim

... 16  17  18  19  20  21  22  23  24  25  [26]  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
12968정성태2/14/20226591오류 유형: 794. msbuild 에러 - error NETSDK1005: Assets file '...\project.assets.json' doesn't have a target for '...'.
12967정성태2/14/20226979VC++: 153. Visual C++ - C99 표준의 Compund Literals 빌드 방법 [4]
12966정성태2/13/20226836.NET Framework: 1155. C# - ffmpeg(FFmpeg.AutoGen): Bitmap으로부터 yuv420p + rawvideo 형식의 파일로 쓰기파일 다운로드1
12965정성태2/13/20226706.NET Framework: 1154. "Hanja Hangul Project v1.01 (파이썬)"의 C# 버전
12964정성태2/11/20227021.NET Framework: 1153. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 avio_reading.c 예제 포팅파일 다운로드1
12963정성태2/11/20227761.NET Framework: 1152. C# - 화면 캡처한 이미지를 ffmpeg(FFmpeg.AutoGen)로 동영상 처리 (저해상도 현상 해결)파일 다운로드1
12962정성태2/9/20227603오류 유형: 793. 마이크로소프트 스토어 - 제품이 존재하지 않습니다. 재고가 없는 것일 수 있습니다.
12961정성태2/8/20227738.NET Framework: 1151. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 프레임의 크기 및 포맷 변경 예제(scaling_video.c) [7]파일 다운로드1
12960정성태2/8/20227162개발 환경 구성: 637. ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 디코딩 예제(decode_video.c) - 세 번째 이야기
12959정성태2/7/20227874.NET Framework: 1150. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 디코딩 예제(decode_video.c) - 두 번째 이야기 [2]파일 다운로드1
12958정성태2/6/20227945.NET Framework: 1149. C# - ffmpeg(FFmpeg.AutoGen) - 비디오 프레임 디코딩 [2]파일 다운로드1
12957정성태2/6/20227571개발 환경 구성: 636. ffmpeg.exe를 이용해 planar 포맷의 데이터를 packed 형식으로 변환하는 방법? [2]
12956정성태2/4/20226780.NET Framework: 1148. C# - ffmpeg(FFmpeg.AutoGen) - decoding 과정 [2]파일 다운로드1
12955정성태2/4/20226175개발 환경 구성: 635. 비주얼 스튜디오에서 실행하던 ASP.NET Core (.NET Framework) 응용 프로그램을 명령행에서 실행하는 방법 (2)
12954정성태2/4/20226029VS.NET IDE: 173. 비주얼 스튜디오 - Output 창에 색상이 지정된 출력 결과가 "[39m[22m" 식의 문자로 나오는 문제
12953정성태2/2/20226279Linux: 48. Windows 11 + WSL 우분투 GUI 환경에서 한글 출력
12952정성태2/2/20226749.NET Framework: 1148. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 오디오 필터 예제(filter_audio.c)파일 다운로드1
12951정성태2/2/20226718.NET Framework: 1147. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 오디오 필터링 예제(filtering_audio.c)파일 다운로드1
12950정성태2/1/20226376.NET Framework: 1146. .NET 6에 추가되지 않은 Generic Math (예: INumber<T>)
12949정성태2/1/20226215.NET Framework: 1145. C# - ffmpeg(FFmpeg.AutoGen) - Codec 정보 열람 및 사용 준비파일 다운로드1
12948정성태1/30/20226338.NET Framework: 1144. C# - ffmpeg(FFmpeg.AutoGen) AVFormatContext를 이용해 ffprobe처럼 정보 출력파일 다운로드1
12947정성태1/30/20227497개발 환경 구성: 634. ffmpeg.exe - 기존 동영상 컨테이너에 다중 스트림을 추가하는 방법
12946정성태1/28/20226012오류 유형: 792. .NET Core - 로컬 개발 중에 docker 호스팅으로 바꾸는 경우 SQL 서버 접근 방법
12945정성태1/28/20226251오류 유형: 791. SQL 서버 로그인 시 localhost는 되고, 127.0.0.1로는 안 되는 문제
12944정성태1/28/20228616.NET Framework: 1143. C# - Entity Framework Core 6 개요
12943정성태1/27/20227531.NET Framework: 1142. .NET 5+로 포팅 시 플랫폼 호환성 경고 메시지(SYSLIB0006, SYSLIB0011, CA1416)파일 다운로드1
... 16  17  18  19  20  21  22  23  24  25  [26]  27  28  29  30  ...