Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 383. BenchmarkDotNet 사용 시 주의 사항 [링크 복사], [링크+제목 복사]
조회: 12435
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

BenchmarkDotNet 사용 시 주의 사항

BenchmarkDotNet으로,

BenchmarkDotNet 라이브러리 소개
; https://www.sysnet.pe.kr/2/0/11547

struct 타입을 좀 더 테스트해봤습니다.

C# 7.2의 특징 - GC 및 메모리 복사 방지를 위한 struct 타입 개선
; https://www.sysnet.pe.kr/2/0/11546

성능 비교를 위해 다음의 타입들을 생성하고,

ClassVector                 : class로 구현
StructVector                : struct로 구현
InStructVector              : StructVector이고, op_Addition 메서드의 인자를 in 처리
ReadonlyInStructVector      : InStructVector이고, readonly struct로 정의
ReadonlyInlineStructVector  : ReadonlyInStructVector이고, op_Addition에 AggressiveInlining 특성 부여

각각의 타입에 대해 op_Addition을 테스트하는 코드만을 Benchmark에 추가했습니다.

public class VectorBenchmark
{
    [Benchmark]
    public void ClassVectorTest()
    {
        var player = new ClassVector(10.0, 20.0, 30.0);
        var speed = new ClassVector(10.0, 20.0, 30.0);
        var result = player + speed;
    }

    [Benchmark]
    public void StructVectorTest()
    {
        var player = new StructVector(10.0, 20.0, 30.0);
        var speed = new StructVector(10.0, 20.0, 30.0);
        var result = player + speed;
    }

    [Benchmark]
    public void InStructVectorTest()
    {
        var player = new InStructVector(10.0, 20.0, 30.0);
        var speed = new InStructVector(10.0, 20.0, 30.0);
        var result = player + speed;
    }

    [Benchmark]
    public void ReadonlyInStructVectorTest()
    {
        var player = new ReadonlyInStructVector(10.0, 20.0, 30.0);
        var speed = new ReadonlyInStructVector(10.0, 20.0, 30.0);
        var result = player + speed;
    }

    [Benchmark]
    public void ReadonlyInlineStructVectorTest()
    {
        var player = new ReadonlyInlineStructVector(10.0, 20.0, 30.0);
        var speed = new ReadonlyInlineStructVector(10.0, 20.0, 30.0);
        var result = player + speed;
    }
}

결과는 이렇습니다.

// * Summary *

BenchmarkDotNet=v0.10.14, OS=Windows 10.0.17134
Intel Core i5-4670 CPU 3.40GHz (Haswell), 1 CPU, 4 logical and 4 physical cores
  [Host]     : .NET Framework 4.7.1 (CLR 4.0.30319.42000), 64bit RyuJIT-v4.7.3101.0
  DefaultJob : .NET Framework 4.7.1 (CLR 4.0.30319.42000), 64bit RyuJIT-v4.7.3101.0


                             Method |       Mean |     Error |    StdDev |
----------------------------------- |-----------:|----------:|----------:|
                    ClassVectorTest | 13.7698 ns | 0.1722 ns | 0.1611 ns |
                   StructVectorTest |  2.5874 ns | 0.0320 ns | 0.0299 ns |
                 InStructVectorTest | 10.5442 ns | 0.0892 ns | 0.0834 ns |
         ReadonlyInStructVectorTest | 10.2489 ns | 0.0843 ns | 0.0788 ns |
     ReadonlyInlineStructVectorTest |  0.5599 ns | 0.0106 ns | 0.0099 ns |

메서드가 inline 처리된 StructVectorTest, ReadonlyInlineStructVectorTest의 성능이 우세한 것은 당연해 보이고, 또한 그 둘 중에서도 복사가 발생하지 않는 ReadonlyInlineStructVectorTest의 성능이 좀 더 높게 나옵니다.

이 상태에서 테스트 코드마다 GetHashCode 호출을 추가했더니,

public void ...VectorTest()
{
    var player = new ...Vector(10.0, 20.0, 30.0);
    var speed = new ...Vector(10.0, 20.0, 30.0);
    var result = player + speed;

    result.GetHashCode();
}

단지 하나의 코드를 추가했을 뿐인데 이제 결과가 완전히 달라집니다.

                             Method |       Mean |     Error |    StdDev |
----------------------------------- |-----------:|----------:|----------:|
                    ClassVectorTest | 34.8412 ns | 0.2300 ns | 0.2151 ns |
                   StructVectorTest | 47.8031 ns | 0.3171 ns | 0.2966 ns |
                 InStructVectorTest | 53.4127 ns | 1.2505 ns | 1.4401 ns |
         ReadonlyInStructVectorTest | 52.1917 ns | 0.3648 ns | 0.3412 ns |
     ReadonlyInlineStructVectorTest | 39.5150 ns | 0.2622 ns | 0.2452 ns |

확인은 해보지 않았지만, GetHashCode 시 값 형식에서 object.GetHashCode 호출로 인한 내부 박싱이 발생해 힙을 사용하도록 변경된 것이 아닌가 예상됩니다. 일단 그건 그렇다 치고, 문제는 모든 Vector 코드에 GetHashCode를 다음과 같이 직접 구현했을 때 발생합니다.

public override int GetHashCode()
{
    return (int)(_x + _y + _z);
}

그럼, 각각의 결과물들이 다음과 같이 바뀝니다.

// GetHashCode() 추가
                             Method |       Mean |     Error |    StdDev |
----------------------------------- |-----------:|----------:|----------:|
                    ClassVectorTest | 15.7360 ns | 0.1863 ns | 0.1651 ns |
                   StructVectorTest | 10.0317 ns | 0.0746 ns | 0.0623 ns |
                 InStructVectorTest | 12.0021 ns | 0.0687 ns | 0.0643 ns |
         ReadonlyInStructVectorTest | 11.8757 ns | 0.1092 ns | 0.1022 ns |
     ReadonlyInlineStructVectorTest |  0.0000 ns | 0.0000 ns | 0.0000 ns |

ReadonlyInlineStructVectorTest의 결과가 놀라운데요, 어떻게 저럴 수 있을까요? 저 때의 JIT 코드를 windbg로 살펴보면 다음과 같이 나옵니다.

0:000> !DumpMD /d 00007ffa11f47790
Method Name:  ConsoleApp1.VectorBenchmark.ReadonlyInlineStructVectorTest()
Class:        00007ffa11f54858
MethodTable:  00007ffa11f477a8
mdToken:      0000000006000007
Module:       00007ffa11f470d0
IsJitted:     yes
CodeAddr:     00007ffa11ea6450
Transparency: Critical

0:000> !U /d 00007ffa11ea6450
Normal JIT generated code
ConsoleApp1.VectorBenchmark.ReadonlyInlineStructVectorTest()
Begin 00007ffa11ea6450, size 1
>>> 00007ffa`11ea6450 c3              ret

보는 바와 같이 정상적인 테스트 코드를 생성하지 못하고 ret만 포함하는 메서드로 테스트하고 있는 것입니다. 따라서 BenchmarkDotNet으로 성능 측정을 할 때는 그 자체의 버그나 Release 모드에서의 최적화에 따른 의도치 않은 코드 삭제를 주의해야 합니다.

게다가 전통적인 Stopwatch를 이용할 때는 Visual Studio 내에서 disassembly 코드를 보며 곧바로 확인할 수 있는 여지가 있는데, BenchmarkDotNet으로 하게 되면 ConsoleApp1.exe의 자식 프로세스로 임시 exe 프로세스를 생성시켜 성능 측정을 하기 때문에 그런 부분을 파헤치기가 불편한 점도 있습니다.




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







[최초 등록일: ]
[최종 수정일: 6/14/2018]

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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12696정성태7/6/20217891VC++: 146. 운영체제의 스레드 문맥 교환(Context Switch)을 유사하게 구현하는 방법파일 다운로드2
12695정성태7/3/20217974VC++: 145. C 언어의 setjmp/longjmp 기능을 Thread Context를 이용해 유사하게 구현하는 방법파일 다운로드1
12694정성태7/2/20219879Java: 24. Azure - Spring Boot 앱을 Java SE(Embedded Web Server)로 호스팅 시 로그 파일 남기는 방법 [1]
12693정성태6/30/20217653오류 유형: 731. Azure Web App Site Extension - Failed to install web app extension [...]. {1}
12692정성태6/30/20217529디버깅 기술: 180. Azure - Web App의 비정상 종료 시 남겨지는 로그 확인
12691정성태6/30/20218349개발 환경 구성: 573. 테스트 용도이지만 테스트에 적합하지 않은 Azure D1 공유(shared) 요금제
12690정성태6/28/20219141Java: 23. Azure - 자바(Java)로 만드는 Web App Service - Tomcat 호스팅
12689정성태6/25/20219709오류 유형: 730. Windows Forms 디자이너 - The class Form1 can be designed, but is not the first class in the file. [1]
12688정성태6/24/20219415.NET Framework: 1073. C# - JSON 역/직렬화 시 리플렉션 손실을 없애는 JsonSrcGen [2]파일 다운로드1
12687정성태6/22/20217434오류 유형: 729. Invalid data: Invalid artifact, java se app service only supports .jar artifact
12686정성태6/21/20219825Java: 22. Azure - 자바(Java)로 만드는 Web App Service - Java SE (Embedded Web Server) 호스팅
12685정성태6/21/202110057Java: 21. Azure Web App Service에 배포된 Java 프로세스의 메모리 및 힙(Heap) 덤프 뜨는 방법
12684정성태6/19/20218499오류 유형: 728. Visual Studio 2022부터 DTE.get_Properties 속성 접근 시 System.MissingMethodException 예외 발생
12683정성태6/18/20219948VS.NET IDE: 166. Visual Studio 2022 - Windows Forms 프로젝트의 x86 DLL 컨트롤이 Designer에서 오류가 발생하는 문제 [1]파일 다운로드1
12682정성태6/18/20217724VS.NET IDE: 165. Visual Studio 2022를 위한 Extension 마이그레이션
12681정성태6/18/20217039오류 유형: 727. .NET 2.0 ~ 3.5 + x64 환경에서 System.EnterpriseServices 참조 시 CS8012 경고
12680정성태6/18/20218135오류 유형: 726. python2.7.exe 실행 시 0xc000007b 오류
12679정성태6/18/20218746COM 개체 관련: 23. CoInitializeSecurity의 전역 설정을 재정의하는 CoSetProxyBlanket 함수 사용법파일 다운로드1
12678정성태6/17/20218008.NET Framework: 1072. C# - CoCreateInstance 관련 Inteop 오류 정리파일 다운로드1
12677정성태6/17/20219439VC++: 144. 역공학을 통한 lxssmanager.dll의 ILxssSession 사용법 분석파일 다운로드1
12676정성태6/16/20219520VC++: 143. ionescu007/lxss github repo에 공개된 lxssmanager.dll의 CLSID_LxssUserSession/IID_ILxssSession 사용법파일 다운로드1
12675정성태6/16/20217556Java: 20. maven package 명령어 결과물로 (war가 아닌) jar 생성 방법
12674정성태6/15/20218318VC++: 142. DEFINE_GUID 사용법
12673정성태6/15/20219476Java: 19. IntelliJ - 자바(Java)로 만드는 Web App을 Tomcat에서 실행하는 방법
12672정성태6/15/202110596오류 유형: 725. IntelliJ에서 Java webapp 실행 시 "Address localhost:1099 is already in use" 오류
12671정성태6/15/202117268오류 유형: 724. Tomcat 실행 시 Failed to initialize connector [Connector[HTTP/1.1-8080]] 오류
... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...