성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] 그냥 RSS Reader 기능과 약간의 UI 편의성 때문에 사용...
[이종효] 오래된 소프트웨어는 보안 위협이 되기도 합니다. 혹시 어떤 기능...
[정성태] @Keystroke IEEE의 문서를 소개해 주시다니... +_...
[손민수 (Keystroke)] 괜히 듀얼채널 구성할 때 한번에 같은 제품 사라고 하는 것이 아...
[정성태] 전각(Full-width)/반각(Half-width) 기능을 토...
[정성태] Vector에 대한 내용은 없습니다. Vector가 닷넷 BCL...
[orion] 글 읽고 찾아보니 디자인 타임에는 InitializeCompon...
[orion] 연휴 전에 재현 프로젝트 올리자 생각해 놓고 여의치 않아서 못 ...
[정성태] 아래의 글에 정리했으니 참고하세요. C# - Typed D...
[정성태] 간단한 재현 프로젝트라도 있을까요? 저런 식으로 설명만 해...
글쓰기
제목
이름
암호
전자우편
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# 9.0 - (15) 최상위 문(Top-level statements)</h1> <p> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C# 9.0 - (1) <a target='tab' href='https://github.com/dotnet/csharplang/issues/100'>대상으로 형식화된 new 식(Target-typed new expressions)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12363'>https://www.sysnet.pe.kr/2/0/12363</a> C# 9.0 - (2) <a target='tab' href='https://github.com/dotnet/csharplang/issues/1738'>localsinit 플래그 내보내기 무시(Suppress emitting localsinit flag)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12364'>https://www.sysnet.pe.kr/2/0/12364</a> C# 9.0 - (3) <a target='tab' href='https://github.com/dotnet/csharplang/issues/111'>람다 메서드의 매개 변수 무시(Lambda discard parameters)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12365'>https://www.sysnet.pe.kr/2/0/12365</a> C# 9.0 - (4) <a target='tab' href='https://github.com/dotnet/csharplang/issues/435'>원시 크기 정수(Native ints)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12366'>https://www.sysnet.pe.kr/2/0/12366</a> C# 9.0 - (5) <a target='tab' href='https://github.com/dotnet/csharplang/issues/1888'>로컬 함수에 특성 지정 가능(Attributes on local functions)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12372'>https://www.sysnet.pe.kr/2/0/12372</a> C# 9.0 - (6) <a target='tab' href='https://github.com/dotnet/csharplang/issues/191'>함수 포인터(Function pointers)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12374'>https://www.sysnet.pe.kr/2/0/12374</a> C# 9.0 - (7) <a target='tab' href='https://github.com/dotnet/csharplang/issues/2850'>패턴 일치 개선 사항(Pattern matching enhancements)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12383'>https://www.sysnet.pe.kr/2/0/12383</a> C# 9.0 - (8) <a target='tab' href='https://github.com/dotnet/csharplang/issues/275'>정적 익명 함수 (static anonymous functions)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12389'>https://www.sysnet.pe.kr/2/0/12389</a> C# 9.0 - (9) <a target='tab' href='https://github.com/dotnet/csharplang/issues/39'>레코드 (Records)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12392'>https://www.sysnet.pe.kr/2/0/12392</a> C# 9.0 - (10) <a target='tab' href='https://github.com/dotnet/csharplang/issues/2460'>대상으로 형식화된 조건식(Target-typed conditional expressions)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12399'>https://www.sysnet.pe.kr/2/0/12399</a> C# 9.0 - (11) <a target='tab' href='https://github.com/dotnet/csharplang/issues/2844'>공변 반환 형식(Covariant return types)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12402'>https://www.sysnet.pe.kr/2/0/12402</a> C# 9.0 - (12) <a target='tab' href='https://github.com/dotnet/csharplang/issues/3194'>foreach 루프에 대한 GetEnumerator 확장 메서드 지원(Extension GetEnumerator)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12403'>https://www.sysnet.pe.kr/2/0/12403</a> C# 9.0 - (13) <a target='tab' href='https://github.com/dotnet/csharplang/blob/master/proposals/csharp-9.0/module-initializers.md'>모듈 이니셜라이저(Module initializers)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12404'>https://www.sysnet.pe.kr/2/0/12404</a> C# 9.0 - (14) <a target='tab' href='https://github.com/dotnet/csharplang/blob/master/proposals/csharp-9.0/extending-partial-methods.md'>부분 메서드에 대한 새로운 기능(New features for partial methods)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12405'>https://www.sysnet.pe.kr/2/0/12405</a> C# 9.0 - (15) <a target='tab' href='https://github.com/dotnet/csharplang/blob/master/proposals/csharp-9.0/top-level-statements.md'>최상위 문(Top-level statements)</a> ; https://www.sysnet.pe.kr/2/0/12406 C# 9.0 - (16) <a target='tab' href='https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-9.0/unconstrained-type-parameter-annotations'>제약 조건이 없는 형식 매개변수 주석(Unconstrained type parameter annotations)</a> ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/12423'>https://www.sysnet.pe.kr/2/0/12423</a> </pre> <br /> <hr style='width: 50%' /><br /> <br /> 근래 타 언어들이 보여주는 REPL(read-eval-print loop) 식 코딩의 편리함을 마침내 C# 9.0부터 언어 자체에서 "Top-level statements"로 도입하게 되었습니다. 이로 인해, 기존에는 간단한 "<a target='tab' href='https://www.helloworld.org/'>Hello World</a>" C# 예제라도 반드시 class와 Main 메서드를 타이핑해야 했는데 이제는 단순히 다음과 같이 입력하고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > System.Console.WriteLine("Hello World!"); </pre> <br /> 실행할 수 있습니다. 물론 내부적으로는 C# 컴파일러의 노력이 있는데요, 사용자가 입력한 실행문을 Program class의 Main 메서드에 넣어 컴파일하는 식으로 처리합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > static class Program { static void Main(string[] args) { // 이곳에 사용자 입력한 실행문을 넣어 빌드 } } </pre> <br /> 기본적으로는 "static void Main"이지만, 사용자가 입력한 코드 유형에 따라 C# 컴파일러는 그에 대응하는 Main 메서드로 처리해 줍니다. 가령 다음과 같이 비동기 await 호출 코드를 포함한다면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > await System.Threading.Tasks.Task.Delay(1000); System.Console.WriteLine("Hi!"); </pre> <br /> "static async Task Main"으로 처리하고, 반환값이 있다면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > await System.Threading.Tasks.Task.Delay(1000); System.Console.WriteLine("Hi!"); return 0; </pre> <br /> "static async Task<int> Main"으로 처리합니다. 여기서 주의할 것은, 기존에 관례적으로 사용하던 Program 타입과 Main 메서드의 이름이 "Top-level statements" 코드를 빌드할 때는 C# 컴파일러에 의해 다른 이름으로 바뀐다는 것입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Console.WriteLine(new StackFrame().GetMethod().DeclaringType.FullName); Console.WriteLine(new StackFrame().GetMethod().Name); /* 출력 결과 <Program>$ <Main>$ */ </pre> <br /> <a target='tab' href='https://www.sysnet.pe.kr/2/0/12392#clone'>"Records"에서 Clone 메서드의 이름</a>이 그랬던 것처럼, Program과 Main의 이름도 유효하지 않은 식별자 이름을 포함하고 있기 때문에 직접적으로 코드에서 그것을 참조해 사용할 수 없습니다.<br /> <br /> <hr style='width: 50%' /><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;' > using System; Device device = new () { Name = "CPU" }; <span style='color: blue; font-weight: bold'>public record Device { public string Name { get; init; } }</span> #if !NET5_0 // .NET 5.0 환경이 아닌 경우 IsExternalInit 클래스를 별도로 정의해서 컴파일 가능하게 만듦 namespace System.Runtime.CompilerServices { public class IsExternalInit { } } #endif </pre> <br /> Main 메서드 내에 넣어서 처리하는 식이 아니고 별도로 (기본 namespace도 없이) 빼서 처리를 합니다.<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) { Device device = new () { Name = "CPU" }; } } <span style='color: blue; font-weight: bold'>public record Device { public string Name { get; init; } }</span> </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;' > using System; const int count = 5; LogText("Count == ", count); <span style='color: blue; font-weight: bold'>static void LogText(string title, object contents) { Console.WriteLine($"{title} {contents}"); }</span> </pre> <br /> Main 메서드 내부에 정의한 로컬 함수로 처리합니다.<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) { const int count = 5; LogText("Count == ", count); <span style='color: blue; font-weight: bold'>static void LogText(string title, object contents) { Console.WriteLine($"{title} {contents}"); }</span> } } </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;' > void PrintName(string name) { } // 컴파일 오류: Error CS0128 A local variable or function named 'PrintName' is already defined in this scope void PrintName(int age) { } </pre> <br /> 게다가 <a target='tab' href='https://www.sysnet.pe.kr/2/0/12372#pinvoke'>pinvoke 구문도 C# 9.0부터 "Attributes on local functions"가 구현</a>되면서 자연스럽게 사용할 수 있게 되었습니다.<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; using System.Runtime.InteropServices; <span style='color: blue; font-weight: bold'>[DllImport("User32.dll", CharSet = CharSet.Unicode)] static extern int MessageBox(IntPtr h, string m, string c, int type);</span> MessageBox(IntPtr.Zero, "C# 9.0", "Top-level statements", 0); </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1655&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> <br /> <hr style='width: 50%' /><br /> <br /> 참고로, 과거에 이와 유사한 기능을 C# 컴파일러인 Roslyn에 ScriptEngine 타입으로 포함시켰고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Roslyn 맛보기 - C# 소스 코드를 스크립트처럼 다루는 방법 ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/1153'>https://www.sysnet.pe.kr/2/0/1153</a> </pre> <br /> 이를 사용해 비주얼 스튜디오에서 Interactive 창까지 포함시켰지요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Roslyn 맛보기 - C# Interactive (1) ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/1154'>https://www.sysnet.pe.kr/2/0/1154</a> Roslyn 맛보기 - C# Interactive (2) ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/1155'>https://www.sysnet.pe.kr/2/0/1155</a> </pre> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
6993
(왼쪽의 숫자를 입력해야 합니다.)