성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Reordering on an Alpha processor ;...
[정성태] 공유 감사합니다. ^^ 참고로, WPF에서 WindowsF...
[Tom Lee] 답변 감사합니다. 나름의 해결책 연구해보고 여기에도 공유해봅니다...
[정성태] 아래의 글을 보면, MoveWindow 하면 될 듯한데요. ^^...
[Tom Lee] 안녕하세요 올려주신 글 참고하여 WPF 어플리케이션 안에 Uni...
[정성태] A graphical depiction of the steps ...
[정성태] 질문을 주셔서 출판사 측에 문의를 했습니다. 약 한 달 정도 후...
[Thorondor
] @정성태 개인 블로그인데도 거의 커뮤니티 급 인 것 같아요. 요...
[정성태] Roll A Lisp In C - Reading ; https...
[정성태] Java - How to use the Foreign Funct...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>OpCodes.Box와 관련해 IL 형식으로 직접 코딩 시 유의할 점</h1> <p> C# 코드에서 object에 값 형식을 전달하는 코드를 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; class Program { static void Main(string[] args) { <span style='color: blue; font-weight: bold'>System.Threading.CancellationToken token</span> = new System.Threading.CancellationToken(); <span style='color: blue; font-weight: bold'>LogOutput(token);</span> // boxing Console.WriteLine("End-of-Test4"); } static void LogOutput(<span style='color: blue; font-weight: bold'>object</span> instance) { Console.WriteLine(instance ?? "null"); } } /* 출력 결과 System.Threading.CancellationToken End-of-Test4 */ </pre> <br /> token 인스턴스가 자연스럽게 object에 대입되는 것처럼 보이지만, 이 과정을 IL 코드로 보면 <a target='tab' href='https://docs.microsoft.com/en-us/dotnet/api/system.reflection.emit.opcodes.box'>box 코드</a>가 관여합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > .method private hidebysig static void Main(string[] args) cil managed { .entrypoint .maxstack 1 .locals init ([0] valuetype [mscorlib]System.Threading.CancellationToken token) IL_0000: nop IL_0001: ldloca.s token IL_0003: initobj [mscorlib]System.Threading.CancellationToken IL_0009: ldloc.0 IL_000a: <span style='color: blue; font-weight: bold'>box [mscorlib]System.Threading.CancellationToken</span> IL_000f: call void Program::LogOutput(object) IL_0014: nop IL_0015: ldstr "End-of-Test4" IL_001a: call void [mscorlib]System.Console::WriteLine(string) IL_001f: nop IL_0020: ret } // end of method Program::Main .method private hidebysig static void LogOutput(object 'instance') cil managed { // ...[생략]... } </pre> <br /> 위의 경우 box 연산자와 함께 "[mscorlib]System.Threading.CancellationToken" 문자열 형식을 전달하는데, 이것은 IL 언어에서 문법적으로 지원하기 때문에 가능한 것이고 만약 바이트를 직접 출력하는 경우라면 다음과 같이 코딩을 해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 0x8c, 0x01000011 // box IL 코드 == 0x8c // // 0x01000011 == System.Threading.CancellationToken 타입에 대한 메타데이터의 토큰값 </pre> <br /> <hr style='width: 50%' /><br /> <br /> 그런데, 여기서 실수를 해 box 연산자를 누락한다면 어떻게 될까요? 실제로 위의 IL 소스 코드에서 box 라인만 주석 처리해,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > .method private hidebysig static void Main(string[] args) cil managed { // ...[생략]... IL_0009: ldloc.0 <span style='color: blue; font-weight: bold'>// IL_000a: box [mscorlib]System.Threading.CancellationToken</span> IL_000f: call void Program::LogOutput(object) IL_0014: nop IL_0015: ldstr "End-of-Test4" IL_001a: call void [mscorlib]System.Console::WriteLine(string) // ...[생략]... } </pre> <br /> ilasm.exe로 exe를 생성/실행하면 화면에는 "(null)"이라는 출력이 나옵니다. 그나마 동작을 하는 듯하지만, 이것은 사실 "예측할 수 없다"는 것이 맞습니다. 이번엔 의도와는 다르게 출력이 되었어도 운이 좋게 프로그램이 종료하지는 않았는데요, 이 코드에서 다음과 같이 직접 정의한 구조체로 바꿔보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; class Program { static void Main(string[] args) { <span style='color: blue; font-weight: bold'>MyStruct token</span> = new MyStruct(); <span style='color: blue; font-weight: bold'>LogOutput(token);</span> Console.WriteLine("End-of-Test4"); } static void LogOutput(<span style='color: blue; font-weight: bold'>object</span> instance) { Console.WriteLine(instance ?? "null"); } } <span style='color: blue; font-weight: bold'>public struct MyStruct</span> { public int Age; public string Name; } </pre> <br /> 이번에는 프로그램이 비정상 종료하면서 이벤트 로그에 다음과 같은 유형의 오류들이 쌓이는 것을 볼 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Log Name: Application Source: .NET Runtime Date: 2018-09-23 오전 11:22:53 Event ID: 1023 Task Category: None Level: Error Keywords: Classic User: N/A Computer: TESTPC Description: Application: console.exe Framework Version: v4.0.30319 Description: The process was terminated due to an internal error in the .NET Runtime at IP 00007FF83A503E30 (00007FF83A500000) with exit code 80131506. </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Log Name: Application Source: Application Error Date: 2018-09-23 오전 11:22:53 Event ID: 1000 Task Category: (100) Level: Error Keywords: Classic User: N/A Computer: TESTPC Description: Faulting application name: console.exe, version: 0.0.0.0, time stamp: 0x5f8660ca Faulting module name: clr.dll, version: 4.8.4250.0, time stamp: 0x5f2a059c Exception code: 0xc0000005 Fault offset: 0x0000000000003e30 Faulting process id: 0x6060 Faulting application start time: 0x01d6a1d0ebe4a979 Faulting application path: C:\temp\bin\Debug\console.exe Faulting module path: C:\Windows\Microsoft.NET\Framework64\v4.0.30319\clr.dll Report Id: a4412515-d89d-462a-9df6-1f428b6ad155 Faulting package full name: Faulting package-relative application ID: </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Log Name: Application Source: Windows Error Reporting Date: 2018-09-23 오전 11:22:55 Event ID: 1001 Task Category: None Level: Information Keywords: Classic User: N/A Computer: TESTPC Description: Fault bucket 1335692000105475064, type 4 Event Name: APPCRASH Response: Not available Cab Id: 0 Problem signature: P1: console.exe P2: 0.0.0.0 P3: 5f8660ca P4: clr.dll P5: 4.8.4250.0 P6: 5f2a059c P7: c0000005 P8: 0000000000003e30 P9: P10: Attached files: \\?\C:\ProgramData\Microsoft\Windows\WER\Temp\WER4D67.tmp.WERInternalMetadata.xml These files may be available here: \\?\C:\ProgramData\Microsoft\Windows\WER\ReportArchive\AppCrash_console.exe_cc693ced9f02ff1fad4971022e1a719ec05145_c6eba193_bcbd42d7-ebe2-4c60-9a33-8dbc2d5cc5ee Analysis symbol: Rechecking for solution: 0 Report Id: a4412515-d89d-462a-9df6-1f428b6ad155 Report Status: 268435456 Hashed bucket: eedc651829064f20028954cc1b9e2bf8 Cab Guid: 0 </pre> <br /> 재현을 하진 못했지만, 어떤 때는 해당 코드를 가진 메서드의 실행 시 "System.Security.VerificationException" 예외가 발생하는 경우도 있습니다.<br /> <br /> 따라서, 값 형식을 object로 넘길 때는 box 연산을 반드시 잊지 않고 사용해야 합니다.<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
4000
(왼쪽의 숫자를 입력해야 합니다.)