Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 383. BenchmarkDotNet 사용 시 주의 사항 [링크 복사], [링크+제목 복사],
조회: 20521
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  [114]  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11075정성태10/20/201621002Windows: 127. Convert-WindowsImage.ps1 사용 방법 정리
11074정성태10/20/201629307Windows: 126. Windows Server 2016 평가판을 정식 버전으로 라이선스 변경하는 방법
11073정성태10/20/201625339.NET Framework: 613. 윈도우 데스크톱 응용 프로그램(예: Console)에서 알림 메시지(Toast notifications) 띄우기 [1]파일 다운로드1
11072정성태10/20/201621877VC++: 102. 새로 추가한 ATL COM 객체가 regsvr32.exe로 등록이 안 되는 문제
11071정성태10/20/201625412.NET Framework: 612. UWP(유니버설 윈도우 플랫폼) 앱에서 콜백 함수 내에서의 UI 요소 접근 방법 [1]
11070정성태10/20/201619623Windows: 125. 윈도우 서버 2016 마이그레이션
11069정성태10/19/201627341.NET Framework: 611. C++ 개발자들을 위한 C# Thread 동작 방식 [2]
11068정성태10/19/201630531Windows: 124. 윈도우 운영체제의 시간 함수 (5) - TSC(Time Stamp Counter)와 QueryPerformanceCounter [12]파일 다운로드1
11067정성태10/18/201626855Windows: 123. 윈도우 운영체제의 시간 함수 (4) - RTC, TSC, PM Clock, HPET Timer [2]
11066정성태10/17/201624500Windows: 122. 윈도우 운영체제의 시간 함수 (3) - QueryInterruptTimePrecise, QueryInterruptTime 함수파일 다운로드1
11065정성태10/15/201629937Windows: 121. 윈도우 운영체제의 시간 함수 (2) - Sleep 함수의 동작 방식 [1]
11064정성태10/14/201621632.NET Framework: 610. C# - WaitOnAddress Win32 API 사용파일 다운로드1
11063정성태10/14/201637741Windows: 120. 윈도우 운영체제의 시간 함수 (1) - GetTickCount와 timeGetTime의 차이점 [5]파일 다운로드1
11062정성태10/12/201618350오류 유형: 361. WCF .svc 호출 시 Could not find a base address that matches scheme net.tcp 예외
11061정성태10/12/201630471오류 유형: 360. IIS - 500.19 오류 (0x80070021)
11060정성태10/12/201622947오류 유형: 359. WCF - .svc 요청시 404 Not Found
11059정성태10/11/201627206.NET Framework: 609. WPF - 다중 스레드 환경에서 데이터 바인딩의 INotifyPropertyChanged.PropertyChanged에 대한 배려 [1]파일 다운로드1
11058정성태10/8/201622531개발 환경 구성: 303. Windows 10 Bash Shell - 한글 환경을 영문으로 바꾸고 싶다면?
11057정성태10/8/201617930오류 유형: 358. Windows 10 bash shell - sudo: unable to resolve host ...
11056정성태10/8/201620790개발 환경 구성: 302. Windows 10 bash shell 시작 시 [...] packages can be updated.
11055정성태10/8/201621839.NET Framework: 608. double 값을 구할 때는 반드시 피연산자를 double로 형변환! [6]
11054정성태10/5/201625856개발 환경 구성: 301. "Let's Encrypt" SSL 인증서를 Azure Cloud Services(classic)에 업데이트하는 방법
11053정성태10/5/201620553.NET Framework: 607. C# try/catch/finally의 IL 코드 표현
11052정성태9/27/201634706개발 환경 구성: 300. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 [7]파일 다운로드1
11051정성태9/25/201621884개발 환경 구성: 299. docker - c:\programdata\docker\windowsfilter 폴더 정리하는 방법파일 다운로드1
11050정성태9/24/201626626VC++: 101. 반올림하지 않고 double 변수 값 출력하는 방법 [3]
... 106  107  108  109  110  111  112  113  [114]  115  116  117  118  119  120  ...