Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법 [링크 복사], [링크+제목 복사]
조회: 11661
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

(시리즈 글이 6개 있습니다.)
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법
; https://www.sysnet.pe.kr/2/0/11563

.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일
; https://www.sysnet.pe.kr/2/0/13162

닷넷: 2175. C# - DllImport 메서드의 AOT 지원을 위한 LibraryImport 옵션
; https://www.sysnet.pe.kr/2/0/13466

닷넷: 2184. C# - 하나의 resource 파일을 여러 프로그램에서 (AOT 시에도) 사용하는 방법
; https://www.sysnet.pe.kr/2/0/13483

개발 환경 구성: 696. C# - 리눅스용 AOT 빌드를 docker에서 수행
; https://www.sysnet.pe.kr/2/0/13487

닷넷: 2202. C# - PublishAot의 glibc에 대한 정적 링킹하는 방법
; https://www.sysnet.pe.kr/2/0/13529




.NET Framework Native compiler 프리뷰 버전 사용법

MVP 내부 그룹에 Native compiler가 preview 상태로 이야기가 오고 가길래 아직 NDA인줄 알았더니 다음과 같이 NuGet에 이미 배포된 버전입니다. ^^

Microsoft.DotNet.Framework.NativeImageCompiler
; https://www.nuget.org/packages/Microsoft.DotNet.Framework.NativeImageCompiler

게다가 문서까지 자세하게 공개되어 있습니다.

Optimize your .NET Desktop apps with native images
; https://go.microsoft.com/fwlink/?linkid=2002310
; https://learn.microsoft.com/en-us/windows/uwp/porting/desktop-to-uwp-r2r

자, 그럼 간단하게 테스트를 해볼까요? ^^

우선, 콘솔 프로젝트를 하나 만들고 다음과 같이 NuGet에서 Native compiler 패키지를 추가해 줍니다.

Install-Package Microsoft.DotNet.Framework.NativeImageCompiler -Version 0.0.1-prerelease-00002 

그다음, 아래의 조건에 맞게 프로젝트의 설정을 변경해 줍니다.

  • .NET Framework 버전은 최소 4.6.2 이상
  • (AnyCPU가 아닌) 명시적으로 x86 또는 x64로 지정
  • Release 모드로 빌드

위와 같은 조건을 만족하면 EXE 파일이 아래의 2개 폴더에 각각 생성됩니다. (유의할 것은, \bin 폴더 하위가 아닙니다.)

\obj\Release\ConsoleApp1.exe
\obj\Release\R2R\ConsoleApp1.exe

자세히 보면 \Release 폴더에 생긴 exe 파일보다 \R2R 폴더에 생긴 EXE 파일의 크기가 살짝 큰 것을 볼 수 있습니다. 왜냐하면, 해당 EXE 파일은 IL 코드 뿐만 아니라 Native 코드까지 함께 포함하고 있기 때문이라고 합니다. 여차하면 IL로 실행하겠다는 의지가 ^^ 엿보이는군요. (따라서, .NET Reflector 같은 역어셈블 도구로 보면 코드가 여전히 잘 보입니다.)

그나저나 Native compiler로 빌드하는 의미가 뭘까요? 단순히, 기계어 컴파일이 되었다는 사실에서 기존 IL 기반의 닷넷 실행 파일보다 "실행 성능"이 빠를 거라고 기대하시면 안 됩니다. 닷넷 프로그램은 Native compiler 처리를 거쳤다고 해서 관리 힙을 쓰지 않는 것은 아닙니다. 즉, 기존의 닷넷 프로그램과 실행 면에서 보면 다를 바가 없습니다. 단지, "JIT Compile" 과정만 생략된 것에 불과합니다.

따라서 JIT 과정이 없어짐으로 인해 초기 로딩 속도가 빨라지는 것이지 "실행 성능" 자체가 좋아지는 것은 아닙니다. 간단하게 예를 들어, 기존에는 소수 판정 프로그램이 1초 걸렸다면 Native compiler로 처리했어도 여전히 1초가 걸리는 것에는 변함이 없습니다. 물론, "최초 소수 판정" 코드의 실행 속도는 빠릅니다. 단지 이후의 동일한 소수 판정 코드가 빨라지는 것은 아닙니다.




참고로, NuGet에서 다운로드 한 Native compiler 실행 파일(NgenR2R.exe)은 다음의 폴더에 위치해 있습니다.

%USERPROFILE%\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools

그런데, 환경에 따라 비주얼 스튜디오에서 빌드 시 다음과 같은 오류가 발생할 수 있습니다.

Error The command "C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe "@obj\Release\\R2R\ConsoleApp1.rsp"" exited with code 9009.

"exited with code 9009" 오류는,

You receive a build error message when you use the Assembly Registration tool (Regasm.exe) in Visual C#
; https://support.microsoft.com/ha-latn-ng/help/908268/you-receive-a-build-error-message-when-you-use-the-assembly-registrati

대개 실행 파일을 찾을 수 없어 발생하는 것 같습니다. 실제로 비주얼 스튜디오가 실행한 하위 프로세스의 명령행을 보면 동일한 이유가 발생했음을 알 수 있습니다.

C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe "@obj\Release\\R2R\ConsoleApp1.rsp"

짐작하셨죠? ^^ 중간에 공백 문자가 있어서 나타난 문제인데 일단 이것을 수정할 방법은 없고 다음과 같이 명령행에서 수작업으로 직접 따옴표를 추가해 실행 파일 경로를 묶어 주면 됩니다.

"C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe" "@obj\Release\\R2R\ConsoleApp1.rsp"

어차피 ^^ preview 버전에 대한 테스트일 뿐이니 딱히 불편한 정도는 아닙니다. 어쨌든, 이렇게 하면 다음과 같이 잘 실행이 됩니다.

E:\ConsoleApp1\ConsoleApp1>"%USERPROFILE%\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe" "@obj\Release\\R2R\ConsoleApp1.rsp"
Microsoft (R) CLR Native Image Generator - Version 4.8.3631.0
Copyright (c) Microsoft Corporation.  All rights reserved.

Loading C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\tools\ref\mscorlib.dll
Loading E:\ConsoleApp1\ConsoleApp1\obj\Release\ConsoleApp1.exe
Native image obj\Release\\R2R\ConsoleApp1.exe generated successfully.




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







[최초 등록일: ]
[최종 수정일: 4/4/2024]

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

비밀번호

댓글 작성자
 



2022-08-16 08시06분
[진우] 이것은 실행시에 CLR 이 필요한 NGEN 인가요?
아니면 NET Native 인가요?
[guest]
2022-08-16 11시25분
.NET Native가 맞습니다. 저도 이때 이후로 제대로 테스트를 못했는데... 진우 님께서 한번 해보시고 결과 좀 공유해 주세요. ^^
정성태

... 16  17  18  19  20  21  [22]  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13080정성태6/17/20226981개발 환경 구성: 643. Visual Studio 2022 17.2 버전에서 C# 11 또는 .NET 7.0 preview 적용
13079정성태6/17/20224644오류 유형: 814. 파이썬 - Error: The file/path provided (...) does not appear to exist
13078정성태6/16/20226692.NET Framework: 2021. WPF - UI Thread와 Render Thread파일 다운로드1
13077정성태6/15/20227019스크립트: 40. 파이썬 - PostgreSQL 환경 구성
13075정성태6/15/20225976Linux: 50. Linux - apt와 apt-get의 차이 [2]
13074정성태6/13/20226295.NET Framework: 2020. C# - NTFS 파일에 사용자 정의 속성값 추가하는 방법파일 다운로드1
13073정성태6/12/20226519Windows: 207. Windows Server 2022에 도입된 WSL 2
13072정성태6/10/20226793Linux: 49. Linux - ls 명령어로 출력되는 디렉터리 색상 변경 방법
13071정성태6/9/20227365스크립트: 39. Python에서 cx_Oracle 환경 구성
13070정성태6/8/20227179오류 유형: 813. Windows 11에서 입력 포커스가 바뀌는 문제 [1]
13069정성태5/26/20229404.NET Framework: 2019. C# - .NET에서 제공하는 3가지 Timer 비교 [2]
13068정성태5/24/20227898.NET Framework: 2018. C# - 일정 크기를 할당하는 동안 GC를 (가능한) 멈추는 방법 [1]파일 다운로드1
13067정성태5/23/20227226Windows: 206. Outlook - 1년 이상 지난 메일이 기본적으로 안 보이는 문제
13066정성태5/23/20226553Windows: 205. Windows 11 - Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
13065정성태5/20/20227231.NET Framework: 2017. C# - Windows I/O Ring 소개 [2]파일 다운로드1
13064정성태5/18/20226758.NET Framework: 2016. C# - JIT 컴파일러의 인라인 메서드 처리 유무
13063정성태5/18/20227181.NET Framework: 2015. C# - 인라인 메서드(inline methods)
13062정성태5/17/20227998.NET Framework: 2014. C# - async/await 그리고 스레드 (4) 비동기 I/O 재현파일 다운로드1
13061정성태5/16/20226803.NET Framework: 2013. C# - FILE_FLAG_OVERLAPPED가 적용된 파일의 읽기/쓰기 시 Position 관리파일 다운로드1
13060정성태5/15/20229284.NET Framework: 2012. C# - async/await 그리고 스레드 (3) Task.Delay 재현파일 다운로드1
13059정성태5/14/20227736.NET Framework: 2011. C# - CLR ThreadPool의 I/O 스레드에 작업을 맡기는 방법 [1]파일 다운로드1
13058정성태5/13/20227644.NET Framework: 2010. C# - ThreadPool.SetMaxThreads 사용법
13057정성태5/12/20229319오류 유형: 812. 파이썬 - ImportError: cannot import name ...
13056정성태5/12/20226452.NET Framework: 2009. C# - async/await 그리고 스레드 (2) MyTask의 호출 흐름 [2]파일 다운로드1
13055정성태5/11/20229383.NET Framework: 2008. C# - async/await 그리고 스레드 (1) MyTask로 재현 [11]파일 다운로드1
13054정성태5/11/20226871.NET Framework: 2007. C# - 10진수 숫자를 담은 문자열을 숫자로 변환하는 방법 [11]파일 다운로드1
... 16  17  18  19  20  21  [22]  23  24  25  26  27  28  29  30  ...