Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2015. C# - 인라인 메서드(inline methods) [링크 복사], [링크+제목 복사],
조회: 15752
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 106  107  [108]  109  110  111  112  113  114  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11224정성태6/13/201718142.NET Framework: 661. Json.NET의 DeserializeObject 수행 시 속성 이름을 동적으로 바꾸는 방법파일 다운로드1
11223정성태6/12/201716824개발 환경 구성: 318. WCF Service Application과 WCFTestClient.exe
11222정성태6/10/201720547오류 유형: 399. WCF - A property with the name 'UriTemplateMatchResults' already exists.파일 다운로드1
11221정성태6/10/201717509오류 유형: 398. Fakes - Assembly 'Jennifer5.Fakes' with identity '[...].Fakes, [...]' uses '[...]' which has a higher version than referenced assembly '[...]' with identity '[...]'
11220정성태6/10/201722899.NET Framework: 660. Shallow Copy와 Deep Copy [1]파일 다운로드2
11219정성태6/7/201718213.NET Framework: 659. 닷넷 - TypeForwardedFrom / TypeForwardedTo 특성의 사용법
11218정성태6/1/201721018개발 환경 구성: 317. Hyper-V 내의 VM에서 다시 Hyper-V를 설치: Nested Virtualization
11217정성태6/1/201716913오류 유형: 397. initerrlog: Could not open error log file 'C:\...\MSSQL12.MSSQLSERVER\MSSQL\Log\ERRORLOG'
11216정성태6/1/201719017오류 유형: 396. Activation context generation failed
11215정성태6/1/201719944오류 유형: 395. 관리 콘솔을 실행하면 "This app has been blocked for your protection" 오류 발생 [1]
11214정성태6/1/201717693오류 유형: 394. MSDTC 서비스 시작 시 -1073737712(0xC0001010) 오류와 함께 종료되는 문제 [1]
11213정성태5/26/201722468오류 유형: 393. TFS - The underlying connection was closed: Could not establish trust relationship for the SSL/TLS secure channel.
11212정성태5/26/201721815오류 유형: 392. Windows Server 2016에 KB4019472 업데이트가 실패하는 경우
11211정성태5/26/201720853오류 유형: 391. BeginInvoke에 전달한 람다 함수에 CS1660 에러가 발생하는 경우
11210정성태5/25/201721294기타: 65. ActiveX 없는 전자 메일에 사용된 "개인정보 보호를 위해 암호화된 보안메일"의 암호화 방법
11209정성태5/25/201768216Windows: 143. Windows 10의 Recovery 파티션을 삭제 및 새로 생성하는 방법 [16]
11208정성태5/25/201727942오류 유형: 390. diskpart의 set id 명령어에서 "The specified type is not in the correct format." 오류 발생
11207정성태5/24/201728235Windows: 142. Windows 10의 복구 콘솔로 부팅하는 방법
11206정성태5/24/201721535오류 유형: 389. DISM.exe - The specified image in the specified wim is already mounted for read/write access.
11205정성태5/24/201721266.NET Framework: 658. C#의 tail call 구현은? [1]
11204정성태5/22/201730804개발 환경 구성: 316. 간단하게 살펴보는 Docker for Windows [7]
11203정성태5/19/201718736오류 유형: 388. docker - Host does not exist: "default"
11202정성태5/19/201719806오류 유형: 387. WPF - There is no registered CultureInfo with the IetfLanguageTag 'ug'.
11201정성태5/16/201722550오류 유형: 386. WPF - .NET 3.5 이하에서 TextBox에 한글 입력 시 TextChanged 이벤트의 비정상 종료 문제 [1]파일 다운로드1
11200정성태5/16/201719324오류 유형: 385. WPF - 폰트가 없어 System.IO.FileNotFoundException 예외가 발생하는 경우
11199정성태5/16/201721149.NET Framework: 657. CultureInfo.GetCultures가 반환하는 값
... 106  107  [108]  109  110  111  112  113  114  115  116  117  118  119  120  ...