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

(시리즈 글이 10개 있습니다.)
.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

디버깅 기술: 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

비밀번호

댓글 작성자
 




... 16  [17]  18  19  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13214정성태1/10/20236006.NET Framework: 2087. .NET 6부터 SourceGenerator와 통합된 System.Text.Json [1]파일 다운로드1
13213정성태1/9/20235482오류 유형: 836. docker 이미지 빌드 시 "RUN apt install ..." 명령어가 실패하는 이유
13212정성태1/8/20235308기타: 85. 단정도/배정도 부동 소수점의 정밀도(Precision)에 따른 형변환 손실
13211정성태1/6/20235299웹: 42. (https가 아닌) http 다운로드를 막는 웹 브라우저
13210정성태1/5/20234406Windows: 219. 윈도우 x64의 경우 0x00000000`7ffe0000 아래의 주소는 왜 사용하지 않을까요?
13209정성태1/4/20234295Windows: 218. 왜 윈도우에서 가상 메모리 공간은 64KB 정렬이 된 걸까요?
13208정성태1/3/20234285.NET Framework: 2086. C# - Windows 운영체제의 2MB Large 페이지 크기 할당 방법파일 다운로드1
13207정성태12/26/20224556.NET Framework: 2085. C# - gpedit.msc의 "User Rights Assignment" 특권을 코드로 설정/해제하는 방법파일 다운로드1
13206정성태12/24/20224806.NET Framework: 2084. C# - GetTokenInformation으로 사용자 SID(Security identifiers) 구하는 방법 [3]파일 다운로드1
13205정성태12/24/20225061.NET Framework: 2083. C# - C++과의 연동을 위한 구조체의 fixed 배열 필드 사용 (2)파일 다운로드1
13204정성태12/22/20224431.NET Framework: 2082. C# - (LSA_UNICODE_STRING 예제로) CustomMarshaler 사용법파일 다운로드1
13203정성태12/22/20224631.NET Framework: 2081. C# Interop 예제 - (LSA_UNICODE_STRING 예제로) 구조체를 C++에 전달하는 방법파일 다운로드1
13202정성태12/21/20225008기타: 84. 직렬화로 설명하는 Little/Big Endian파일 다운로드1
13201정성태12/20/20225560오류 유형: 835. PyCharm 사용 시 C 드라이브 용량 부족
13200정성태12/19/20224459오류 유형: 834. 이벤트 로그 - SSL Certificate Settings created by an admin process for endpoint
13199정성태12/19/20224699개발 환경 구성: 656. Internal Network 유형의 스위치로 공유한 Hyper-V의 VM과 호스트가 통신이 안 되는 경우
13198정성태12/18/20224622.NET Framework: 2080. C# - Microsoft.XmlSerializer.Generator 처리 없이 XmlSerializer 생성자를 예외 없이 사용하고 싶다면?파일 다운로드1
13197정성태12/17/20224489.NET Framework: 2079. .NET Core/5+ 환경에서 XmlSerializer 사용 시 System.IO.FileNotFoundException 예외 발생하는 경우파일 다운로드1
13196정성태12/16/20224677.NET Framework: 2078. .NET Core/5+를 위한 SGen(Microsoft.XmlSerializer.Generator) 사용법
13195정성태12/15/20225161개발 환경 구성: 655. docker - bridge 네트워크 모드에서 컨테이너 간 통신 시 --link 옵션 권장 이유
13194정성태12/14/20225235오류 유형: 833. warning C4747: Calling managed 'DllMain': Managed code may not be run under loader lock파일 다운로드1
13193정성태12/14/20225339오류 유형: 832. error C7681: two-phase name lookup is not supported for C++/CLI or C++/CX; use /Zc:twoPhase-
13192정성태12/13/20225380Linux: 55. 리눅스 - bash shell에서 실수 연산
13191정성태12/11/20226282.NET Framework: 2077. C# - 직접 만들어 보는 SynchronizationContext파일 다운로드1
13190정성태12/9/20226783.NET Framework: 2076. C# - SynchronizationContext 기본 사용법파일 다운로드1
13189정성태12/9/20227543오류 유형: 831. Visual Studio - Windows Forms 디자이너의 도구 상자에 컨트롤이 보이지 않는 문제
... 16  [17]  18  19  20  21  22  23  24  25  26  27  28  29  30  ...