Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법 [링크 복사], [링크+제목 복사]
조회: 11627
글쓴 사람
정성태 (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가 맞습니다. 저도 이때 이후로 제대로 테스트를 못했는데... 진우 님께서 한번 해보시고 결과 좀 공유해 주세요. ^^
정성태

... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12602정성태4/16/20219478VS.NET IDE: 161. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 [1]
12601정성태4/15/20218560.NET Framework: 1040. C# - REST API 대신 github 클라이언트 라이브러리를 통해 프로그래밍으로 접근
12600정성태4/15/20218752.NET Framework: 1039. C# - Kubeconfig의 token 설정 및 인증서 구성을 자동화하는 프로그램
12599정성태4/14/20219419.NET Framework: 1038. C# - 인증서 및 키 파일로부터 pfx/p12 파일을 생성하는 방법파일 다운로드1
12598정성태4/14/20219581.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)파일 다운로드1
12597정성태4/13/20219634개발 환경 구성: 569. csproj의 내용을 공통 설정할 수 있는 Directory.Build.targets / Directory.Build.props 파일
12596정성태4/12/20219380개발 환경 구성: 568. Windows의 80 포트 점유를 해제하는 방법
12595정성태4/12/20218814.NET Framework: 1036. SQL 서버 - varbinary 타입에 대한 문자열의 CAST, CONVERT 변환을 C# 코드로 구현
12594정성태4/11/20218258.NET Framework: 1035. C# - kubectl 명령어 또는 REST API 대신 Kubernetes 클라이언트 라이브러리를 통해 프로그래밍으로 접근 [1]파일 다운로드1
12593정성태4/10/20219397개발 환경 구성: 567. Docker Desktop for Windows - kubectl proxy 없이 k8s 대시보드 접근 방법
12592정성태4/10/20219205개발 환경 구성: 566. Docker Desktop for Windows - k8s dashboard의 Kubeconfig 로그인 및 Skip 방법
12591정성태4/9/202112453.NET Framework: 1034. C# - byte 배열을 Hex(16진수) 문자열로 고속 변환하는 방법 [2]파일 다운로드1
12590정성태4/9/20218967.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202110305.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
12588정성태4/7/202110884개발 환경 구성: 565. PowerShell - New-SelfSignedCertificate를 사용해 CA 인증서 생성 및 인증서 서명 방법
12587정성태4/6/202111658개발 환경 구성: 564. Windows 10 - ClickOnce 배포처럼 사용할 수 있는 MSIX 설치 파일 [1]
12586정성태4/5/20219346오류 유형: 710. Windows - Restart-Computer / shutdown 명령어 수행 시 Access is denied(E_ACCESSDENIED)
12585정성태4/5/20219092개발 환경 구성: 563. 기본 생성된 kubeconfig 파일의 내용을 새롭게 생성한 인증서로 구성하는 방법
12584정성태4/1/20219791개발 환경 구성: 562. kubeconfig 파일 없이 kubectl 옵션만으로 실행하는 방법
12583정성태3/29/202111322개발 환경 구성: 561. kubectl 수행 시 다른 k8s 클러스터로 접속하는 방법
12582정성태3/29/202110002오류 유형: 709. Visual C++ - 컴파일 에러 error C2059: syntax error: '__stdcall'
12581정성태3/28/20219950.NET Framework: 1031. WinForm/WPF에서 Console 창을 띄워 출력하는 방법 (2) - Output 디버깅 출력을 AllocConsole로 우회 [2]
12580정성태3/28/20218671오류 유형: 708. SQL Server Management Studio - Execution Timeout Expired.
12579정성태3/28/20218709오류 유형: 707. 중첩 가상화(Nested Virtualization) - The virtual machine could not be started because this platform does not support nested virtualization.
12578정성태3/27/20219043개발 환경 구성: 560. Docker Desktop for Windows 기반의 Kubernetes 구성 (2) - WSL 2 인스턴스에 kind가 구성한 k8s 서비스 위치
12577정성태3/26/202111121개발 환경 구성: 559. Docker Desktop for Windows 기반의 Kubernetes 구성 - WSL 2 인스턴스에 kind 도구로 k8s 클러스터 구성
... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...