성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
[정성태] "props[i].GetValue(props[i])" 코드에서 ...
글쓰기
제목
이름
암호
전자우편
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'>[부연 설명] 쉬운 C# 코드를 어럽게 이해하기</h1> <p> 이 글은 다음의 글에 대한 릴레이 성격의 글이라고 봐주시면 되겠습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 쉬운 C# 코드를 어럽게 이해하기 ; <a target='tab' href='http://www.csharpstudy.com/Network/DevNote/Article/20'>http://www.csharpstudy.com/Network/DevNote/Article/20</a> </pre> <br /> 위의 글에 설명이 모두 맞지만, 약간의 곡해를 하실 분들이 있을 것 같아 좀 더 부가적인 설명이 필요하지 않을까 싶어 글을 쓰게 되었습니다.<br /> <br /> <br /><div style='font-size: 12pt; font-family: Malgun Gothic, Consolas; color: #2211AA; text-align: left; font-weight: bold'>1. 빌드된 MyTest.EXE는 일반 Win32 프로그램과 다른 PE 헤더를 가지고 있으며 </div><br /> <br /> 위에서 언급한 "다른 PE 헤더"는 헤더의 구조가 다르다는 것이 아닙니다. 닷넷 EXE와 Win32의 EXE는 PE 헤더 구조가 동일합니다. 다르다고 하는 부분은 PE 헤더 중에서 IMAGE_OPTIONAL_HEADER 내부의 DataDirectory 구조체 중의 하나인 CLRRuntimeHeader가 비어있느냐, 채워져 있느냐 하는 부분입니다.<br /> <br /> 이와 관련된 보다 자세한 설명은 다음의 글을 참고하세요.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/1296'>http://www.sysnet.pe.kr/2/0/1296</a> </pre> <br /> <br /><div style='font-size: 12pt; font-family: Malgun Gothic, Consolas; color: #2211AA; text-align: left; font-weight: bold'>2. 하나의 Managed 프로세스는 하나의 CLR을 갖는다</div> <br /> 이 문장은 꼭 그 다음에 나오는 "각각의 프로세스에 1개씩 총 10개의 clr.dll 이 Physically 존재한다"와 함께 연관시켜 해석해야 합니다. 즉 "하나의 (.NET 4.0) Managed 프로세스는 하나의 CLR.dll을 갖는다."로 이해하셔야 합니다.<br /> <br /> 사실 닷넷은 하나의 EXE 내에 side-by-side로 로드할 수 있는 기능이 있습니다. 다시 말해 CLR 2, CLR 4를 모두 하나의 EXE에서 호스팅할 수 있습니다.<br /> <br /> 가령, 아래의 SampleBrowser 응용 프로그램에서 "CppHostCLR" 소스코드를 다운로드 받아 실행해 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C++ app hosts CLR 4 and invokes .NET assembly (CppHostCLR) ; <a target='tab' href='http://1code.codeplex.com/downloads/get/clickOnce/SampleBrowser.application'>http://1code.codeplex.com/downloads/get/clickOnce/SampleBrowser.application</a> </pre> <br /> 하나의 EXE 내에 CLR 2, 4가 함께 올라오고 따라서 CLR 2용의 mscorwks.dll, CLR 4용의 clr.dll을 확인할 수 있습니다.<br /> <br /> <br /><div style='font-size: 12pt; font-family: Malgun Gothic, Consolas; color: #2211AA; text-align: left; font-weight: bold'>3. CLR은 EXE의 Entry Point인 Main() 메서드를 호출하게 되는데, Main 메서드는 어느 클래스에나 존재할 수 있지만 반드시 Static 이어야 하고 프로그램 내에 단 1개만 존재해야 한다.</div> <br /> 이것은 EXE에서 최초 실행되는 메서드가 하나만 지정될 수 있다는 것이지, static Main 메서드가 프로그램 내에 1개만 존재할 수 있다는 것은 아닙니다.<br /> <br /> 가령 다음과 같이 2개의 클래스에 Main 메서드를 정의해도 무방합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > class Program { static void Main(string[] args) { } } class Test { static void Main(string[] args) { } } </pre> <br /> 이런 경우 C# 컴파일러가 자동으로 하나의 Main 메서드를 선택할 수 없기 때문에 오류가 나는 것인데요. Visual Studio의 경우 프로젝트 속성 창에서 다음과 같이 어떤 Main을 선택할 지 지정해 주면 됩니다.<br /> <br /> <img alt='cs_basic_1.png' src='/SysWebRes/bbs/cs_basic_1.png' /> <br /><br /> 그런 경우 Visual Studio는 csc.exe 컴파일러의 인자에 "/main:<type>" 옵션을 함께 전달해 컴파일합니다.<br /> <br /> <br /><div style='font-size: 12pt; font-family: Malgun Gothic, Consolas; color: #2211AA; text-align: left; font-weight: bold'>4. Managed Heap에 MyClass 객체 메모리를 할당한 후, 쓰레드는 c 라는 로컬 변수를 해당 쓰레드의 스택에 4 byte만큼 생성하고 </div><br /> <br /> 추상화된 관점에서 이해한다는 차원에서는 이 설명으로 넘어가셔도 됩니다. 단지, 좀 더 엄밀하게 들어가면 순서가 다릅니다.<br /> <br /> (이후 x86 기준으로 설명합니다.) C# 컴파일러는 해당 메서드를 컴파일하는 단계에서 그 메서드의 실행에 필요한 스택 크기를 이미 계산해 냅니다. 그렇게 계산한 스택 크기는 메서드의 실행 진입점에서 할당됩니다. 일례로 IL 코드의 형식을 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > .class private auto ansi beforefieldinit Program extends [mscorlib]System.Object { .method private hidebysig static void Main(string[] args) cil managed { .entrypoint <span style='color: blue; font-weight: bold'>.maxstack 8</span> IL_0000: nop IL_0001: ldstr "Hello World" IL_0006: call void [mscorlib]System.Console::WriteLine(string) IL_000b: nop IL_000c: ret } } </pre> <br /> 이렇게 그 메서드를 실행하기 위한 스택의 크기가 이미 지정되어 있습니다. 이를 반영해 JIT 컴파일러는 네이티브 코드로 스택 프레임을 형성한 후 로컬 변수를 위한 공간을 메서드 진입 시점에 이미 생성해 놓습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > void Test() { push ebp mov ebp,esp <span style='color: blue; font-weight: bold'>sub esp,0Ch // 쓰레드는 메서드 진입 단계에서 스택 크기를 이미 생성하고,</span> // 이후 new MyClass();가 나오면 Managed Heap에 MyClass 객체를 할당하고, // 힙에 할당된 객체의 주소를 가리키는 주소를 이전에 sub esp, 0ch로 확보한 스택에 보관 } </pre> <br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
2018
(왼쪽의 숫자를 입력해야 합니다.)