Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2015. C# - 인라인 메서드(inline methods) [링크 복사], [링크+제목 복사],
조회: 15731
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

(시리즈 글이 11개 있습니다.)
.NET Framework: 202. CLR JIT 컴파일러가 생성한 기계어 코드 확인하는 방법
; https://www.sysnet.pe.kr/2/0/975

.NET Framework: 210. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후
; https://www.sysnet.pe.kr/2/0/1023

.NET Framework: 395. C# - 프로퍼티로 정의하면 필드보다 느릴까요?
; https://www.sysnet.pe.kr/2/0/1545

.NET Framework: 396. C# - 프로퍼티로 정의하면 필드보다 느릴까요? - windbg / ollydbg
; https://www.sysnet.pe.kr/2/0/1546

.NET Framework: 542. 닷넷 - 특정 클래스가 로드되었는지 여부를 알 수 있을까?
; https://www.sysnet.pe.kr/2/0/10888

.NET Framework: 545. 닷넷 - 특정 클래스가 로드되었는지 여부를 알 수 있을까? - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/10893

.NET Framework: 763. .NET Core 2.1 - Tiered Compilation 도입
; https://www.sysnet.pe.kr/2/0/11539

디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/12133

.NET Framework: 2015. C# - 인라인 메서드(inline methods)
; https://www.sysnet.pe.kr/2/0/13063

.NET Framework: 2016. C# - JIT 컴파일러의 인라인 메서드 처리 유무
; https://www.sysnet.pe.kr/2/0/13064

닷넷: 2132. C# - sealed 클래스의 메서드를 callback 호출했을 때 인라인 처리가 될까요?
; https://www.sysnet.pe.kr/2/0/13391




C# - 인라인 메서드(inline methods)

지인 한 분이 아래의 글이 뭘 설명하는지 잘 모르겠다고 ^^ 질문을 하셨습니다.

.NET Core Best Practices
; https://www.nilebits.com/blog/2022/04/net-core-best-practices/

By passing arguments, reducing jumps, and restoring registers, inline methods improve app performance. Remember that the JIT (just-in-time) compiler will not inline one method that contains a throw statement. To fix it, create a static helper process that includes a throw statement.


사실 C/C++ 언어를 해보신 분들에게는 자연스러운 듯한 문장인데 아무래도 C#과 같은 VM 언어만을 다뤄본 분들에게는 어려울 수 있을 듯합니다.

간단하게 풀어볼까요? 이를 위해 위의 문장을 2개로 나누고,

[1]
By passing arguments, reducing jumps, and restoring registers, inline methods improve app performance.

[2]
Remember that the JIT (just-in-time) compiler will not inline one method that contains a throw statement. To fix it, create a static helper process that includes a throw statement.


그중에서 "[1]"에 해당하는 부분을 다음의 코드를 곁들여 설명해 보겠습니다.

int x = 0;
int result = Increment(x);
Console.WriteLine(result);

int Increment(int x)
{
    return (x + 1);
}

보는 바와 같이 Increment 메서드를 호출하고 있는데요, 메서드로 구현은 되었지만 코드 내용이 간단하다는 특징이 있습니다. 이런 경우 최적화 과정에서 컴파일러(닷넷의 경우 JIT 컴파일러)에 의해 다음과 같이 함수 호출이 아닌, 말 그대로 호출 측으로 인라인 되는 것도 가능합니다.

int x = 0;
int result = x + 1;
Console.WriteLine(result);

즉, 컴파일러의 재량에 의해 전체적인 코드 수행에는 논리적인 오류 없이 코드를 inline 시키는 것입니다. 그렇다면 컴파일러는 왜? 이런 고생을 사서 하는 것일까요? ^^

사실 함수 호출은 어찌 보면 꽤나 비용이 드는 작업입니다. 왜냐하면, 호출자와 피호출자 측에서는 다음과 같은 작업을 요구하기 때문입니다.

[호출자]
인자를 전달하기 위해 스택 또는 레지스터에 push

[피호출자]
스택 프레임, non-volatile 레지스터 백업 등의 prologue 코드
...본 코드 수행...
return 전, 스택 프레임 및 non-volatile 레지스터 등의 복원을 수행하는 epilogue 코드

일반적인 경우라면 Increment 메서드 호출은 저런 코드 수행을 모두 동반해도 문제가 없지만, 만약 반복이 많은 루프 문 내에서 사용되는 경우라면 인라인 메서드의 사용만으로 반복문 수행의 성능을 크게 높일 수 있다는 장점이 있습니다.

이것으로 "[1]"번 부분은 이해하시겠죠? ^^ 자, 그럼 "[2]"번 문장으로 가볼까요?

[2]
Remember that the JIT (just-in-time) compiler will not inline one method that contains a throw statement. To fix it, create a static helper process that includes a throw statement.


컴파일러의 인라인 처리가 최적화의 일환으로 수행은 되지만, 그렇다고 모든 유형의 메서드를 인라인 시킬 수 있는 것입니다. 몇몇 제약이 있는데요, 그중 하나가 바로 "throw" 문을 포함하는 메서드는 인라인 시킬 수 없다는 것입니다. 가령 Increment 메서드를 이렇게 바꾸면,

int Increment(int x)
{
    if (x < 0)
    {
        throw new ArgumentException();
    }

    return (x + 1);
}

이제 JIT 컴파일러는 Increment 메서드를 throw 문의 포함으로 인해 인라인 시킬 수 없습니다. 대신, 아래와 같이 throw 문을 직접 포함하지 않는 호출로 바꾸면,

static int Increment(int x)
{
    if (x < 0)
    {
        ThrowArgumentException();
    }

    return (x + 1);
}

static int ThrowArgumentException()
{
    throw new ArgumentException();
}

다시, Increment의 내용을 인라인 시키는 것이 가능하다는 것입니다. (그런데, 정말 위의 설명이 사실일까요? 이에 대해서는 다음번 글에서 알아보겠습니다. ^^)




참고로, 인라인 메서드의 좋은 예가 바로 C# 프로퍼티 구문입니다.

C# - 프로퍼티로 정의하면 필드보다 느릴까요?
; https://www.sysnet.pe.kr/2/0/1545

잘 아시는 것처럼, get/set 구문은 결국 메서드로 처리됩니다. 하지만, JIT 컴파일러는 get/set 메서드의 코드가 간결한 경우 인라인 처리를 해버리기 때문에 메서드 호출에 대한 부하가 없습니다. 그러니, 안심하고 마음껏 (필드가 아닌) 프로퍼티로 정의해도 됩니다. ^^




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







[최초 등록일: ]
[최종 수정일: 9/3/2023]

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

비밀번호

댓글 작성자
 




... 151  152  153  154  155  156  157  158  159  [160]  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1048정성태5/27/201132245개발 환경 구성: 123. Apache 소스를 윈도우 환경에서 빌드하기
1047정성태5/27/201126115.NET Framework: 217. Firebird ALinq Provider - 날짜 필드에 대한 낙관적 동시성 쿼리 오류
1046정성태5/26/201130780.NET Framework: 216. 라이선스까지도 뛰어넘는 .NET Profiler [5]
1045정성태5/24/201131864.NET Framework: 215. 닷넷 System.ComponentModel.LicenseManager를 이용한 라이선스 적용 [1]파일 다운로드1
1044정성태5/24/201132425오류 유형: 122. zlib 빌드 오류 - inflate.obj : error LNK2001: unresolved external symbol _inflate_fast
1043정성태5/24/201131402.NET Framework: 214. 무료 Linq Provider - DbLinq를 이용한 Firebird 접근파일 다운로드1
1042정성태5/23/201137731개발 환경 구성: 122. PHP 소스를 윈도우 환경에서 빌드하기
1041정성태5/22/201128625.NET Framework: 213. Linq To SQL - ALinq Provider를 이용하여 Firebird 사용파일 다운로드1
1040정성태5/21/201138947개발 환경 구성: 121. .NET 개발자가 처음 설치해 본 Apache + PHP [2]
1039정성태5/17/201131654.NET Framework: 212. Firebird 데이터베이스와 ADO.NET [2]파일 다운로드1
1038정성태5/16/201133618개발 환경 구성: 120. .NET 프로그래머에게도 유용한 Firebird 무료 데이터베이스 [2]
1037정성태5/11/201128461개발 환경 구성: 119. Visual Studio Professional 이하 버전에서도 TFS의 정적 코드 분석 정책 연동이 가능할까? [3]
1036정성태5/7/201194266오류 유형: 121. Access DB에 대한 32bit/64bit OLE DB Provider 관련 오류 [11]
1035정성태5/7/201129018오류 유형: 120. File cannot be opened. Ensure it is a valid Data Link file.
1034정성태5/2/201126056.NET Framework: 211. 파일 잠금 없이 .NET 어셈블리의 버전을 구하는 방법 [2]파일 다운로드1
1033정성태5/1/201131767웹: 19. IIS Express - appcmd.exe를 이용한 applicationHost.config 변경 [2]
1032정성태5/1/201128414웹: 18. IIS Express를 NT 서비스로 변경
1031정성태4/30/201129556웹: 17. IIS Express - "IIS Installed Versions Manager Interface"의 IIISExpressProcessUtility 구하는 방법 [1]파일 다운로드1
1030정성태4/30/201151846개발 환경 구성: 118. IIS Express - localhost 이외의 호스트 이름으로 접근하는 방법 [4]파일 다운로드1
1029정성태4/28/201140977개발 환경 구성: 117. XCopy에서 파일/디렉터리 확인 질문 없애기 [2]
1028정성태4/27/201138366오류 유형: 119. Visual Studio 2010 SP1 설치 후 Windows Phone 개발자 도구로 인한 재설치 문제 [3]
1027정성태4/25/201127541디버깅 기술: 40. 상황별 GetFunctionPointer 반환값 정리 - x86파일 다운로드1
1026정성태4/25/201145834디버깅 기술: 39. DebugDiag 1.1을 사용한 덤프 분석 [7]
1025정성태4/24/201127893개발 환경 구성: 116. IIS 7 관리자 - Active Directory Certification Authority로부터 SSL 사이트 인증서 받는 방법 [2]
1024정성태4/22/201129197오류 유형: 118. Windows 2008 서버에서 Event Viewer / PowerShell 실행 시 비정상 종료되는 문제 [1]
1023정성태4/20/201130082.NET Framework: 210. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 [1]
... 151  152  153  154  155  156  157  158  159  [160]  161  162  163  164  165  ...