Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2015. C# - 인라인 메서드(inline methods) [링크 복사], [링크+제목 복사],
조회: 15728
글쓴 사람
정성태 (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)
1227정성태2/3/201229268.NET Framework: 299. 해당 어셈블리가 Debug 빌드인지, Release 빌드인지 알아내는 방법파일 다운로드1
1226정성태1/28/201270198.NET Framework: 298. 홀 펀칭(Hole Punching)을 이용한 Private IP 간 통신 - C# [15]파일 다운로드3
1225정성태1/24/201225854.NET Framework: 297. 특정 EXE 파일의 실행을 Internet Explorer처럼 "Protected Mode"로 실행하는 방법 [1]파일 다운로드1
1224정성태1/21/201237346개발 환경 구성: 139. 아마존 EC2에 새로 추가된 "1년 무료 Windows 서버 인스턴스"가 있다는데, 직접 만들어 볼까요? ^^ [11]
1223정성태1/20/201227312.NET Framework: 296. 괜찮은 문자열 해시함수? - 두 번째 이야기 [1]파일 다운로드1
1222정성태1/18/201235032.NET Framework: 295. 괜찮은 문자열 해시 함수? [4]파일 다운로드1
1221정성태1/17/201224024오류 유형: 147. System.Runtime.InteropServices.COMException (0x80005000)
1220정성태1/15/201224216.NET Framework: 294. Master web.config 파일을 수정하려면?파일 다운로드1
1219정성태1/15/201226582.NET Framework: 293. Microsoft PowerPoint 슬라이드를 HTML 파일로 ".files" 폴더 없이 저장하는 방법 (C# 코드)파일 다운로드1
1218정성태1/15/201239136.NET Framework: 292. RSACryptoServiceProvider의 공개키와 개인키 구분 [1]파일 다운로드2
1217정성태1/14/201241230.NET Framework: 291. .NET에서 WAV, MP3 파일 재생하는 방법 [1]파일 다운로드1
1216정성태1/14/201229942오류 유형: 146. Microsoft Visual C++ 재배포 패키지 - 설치 로그 남기는 방법 [1]
1215정성태1/9/201227497제니퍼 .NET: 20. 제니퍼 닷넷 적용 사례 (3) - '닷넷'이 문제일까? '닷넷 개발자'가 문제일까? [6]
1214정성태1/3/201224326제니퍼 .NET: 19. 제니퍼 닷넷 설치/제거 방법 - IIS
1213정성태12/31/201124284.NET Framework: 290. WCF - 접속된 클라이언트의 IP 주소 알아내는 방법 - 두 번째 이야기
1212정성태12/31/201124365오류 유형: 145. The trust relationship between this workstation and the primary domain failed.
1211정성태12/31/201129152.NET Framework: 289. WindowsFormsHost를 사용하는 XBAP 응용 프로그램파일 다운로드1
1210정성태12/30/201148127.NET Framework: 288. FFmpeg.exe를 이용한 C# 동영상 인코더 예제 [9]파일 다운로드1
1209정성태12/29/201122770개발 환경 구성: 138. BizTalk 2006 설치 방법
1208정성태12/28/201145769.NET Framework: 287. Excel Sheet를 WinForm에서 사용하는 방법 [8]파일 다운로드2
1207정성태12/26/201125042.NET Framework: 286. x86/x64로 구분된 코드를 포함하는 경우, 다중으로 어셈블리를 만들어야 할까요?파일 다운로드1
1206정성태12/25/201126056.NET Framework: 285. Shader 강좌와 함께 배워보는 XNA Framework (3) - 텍스처 매핑 예제파일 다운로드1
1205정성태12/25/201131704.NET Framework: 284. Thread 개체의 Interrupt와 Abort의 차이점파일 다운로드1
1204정성태12/22/201125203.NET Framework: 283. MEF를 ASP.NET에 성능 손실 없이 적용하려면? [7]
1203정성태12/21/201125574제니퍼 .NET: 18. MEF가 적용된 ASP.NET 웹 사이트를 제니퍼 닷넷으로 모니터링 해본 결과! [6]
1202정성태12/21/201126002오류 유형: 144. The database '...' cannot be opened because it is version 661.
... 151  152  [153]  154  155  156  157  158  159  160  161  162  163  164  165  ...