성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
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# 12 - 기본 람다 매개 변수</h1> <p> ("람다 식의 선택적 매개 변수"라고 예전에 제목을 지었는데 <a target='tab' href='https://learn.microsoft.com/ko-kr/dotnet/csharp/whats-new/csharp-12#default-lambda-parameters'>공식 문서</a>에 "기본 람다 매개 변수"로 나와서 그에 따라 이름을 바꿉니다.)<br /> <br /> <hr style='width: 50%' /><br /> <br /> C# Next, 즉 C# 12에 포함될 기능에 대한 진행 사항이 아래의 문서를 통해 공개돼 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C# Next ; <a target='tab' href='https://github.com/dotnet/roslyn/blob/main/docs/Language%20Feature%20Status.md#c-next'>https://github.com/dotnet/roslyn/blob/main/docs/Language%20Feature%20Status.md#c-next</a> .NET Conf 2023 - "What's New in C# 12" ; <a target='tab' href='https://youtu.be/xEFO1sQ2bUc?t=16316'>https://youtu.be/xEFO1sQ2bUc?t=16316</a> </pre> <br /> 12개의 개선이 추가되는데 오늘은 그 첫 번째로 "<a target='tab' href='https://github.com/dotnet/csharplang/issues/6051'>Lambda default parameters</a>"를 소개합니다. ^^ 이것을 먼저 선택한 이유는, 해당 기능은 현재 버전의 Visual Studio 2022 17.5.5 버전에서 테스트할 수 있기 때문입니다. (다른 기능들은 Preview 버전의 Visual Studio 2022를 요구합니다.)<br /> <br /> <a name='lang_version'></a> 따라서 이 글의 실습을 하려면 프로젝트 생성 후 LangVersion만 preview로 바꾸면 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <Project Sdk="Microsoft.NET.Sdk"> <PropertyGroup> <OutputType>Exe</OutputType> <TargetFramework>net7.0</TargetFramework> <ImplicitUsings>enable</ImplicitUsings> <Nullable>enable</Nullable> <span style='color: blue; font-weight: bold'><LangVersion>preview</LangVersion></span> </PropertyGroup> </Project> </pre> <br /> <hr style='width: 50%' /><br /> <br /> C#에서 발전 중인 Lambda의 역사는 다음과 같이 정리할 수 있습니다<br /> <br /> <ul> <li>C# 3.0: 처음 람다 지원</li> <li>C# 6.0: 람다 표현식을 이용해 일반 메서드, 속성과 인덱서의 get 접근자에 대한 정의</li> <li>C# 7.0: C# 6.0에 구현된 범위를 확장 - 생성자, 종료자, 이벤트의 add/remove, 속성과 인덱서의 set 접근자</li> <li>C# 9.0: 매개변수 무시, static 지원</li> <li>C# 10: 특성 허용, 반환 타입 지정, var 추론</li> </ul> <br /> 뭐랄까, 일반 메서드에서 허용된 기능들이 람다에서도 지원되고 있는 건데요, 사실 람다뿐만 아니라 익명 메서드, 로컬 함수까지 전방위적으로 지속적인 개선이 이뤄지고 있습니다.<br /> <br /> 그리고 C# 12에는 람다에 대해 (일반 메서드에나 가능했던) 매개변수의 기본값까지 설정하는 것이 가능해졌습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > var addWithDefault = (<span style='color: blue; font-weight: bold'>int addTo = 2</span>) => addTo + 1; addWithDefault(); // 3 addWithDefault(5); // 6 </pre> <br /> 여기서 재미있는 것은, 람다 자체를 받는 것은 결국 delegate 타입이기 때문에 사실상 기본값에 대한 설정은 람다가 아닌 delegate에 있다는 점입니다. 단지, 위의 코드는 var가 그것을 추론해 대응하는 delegate 타입을 C# 컴파일러가 대신 만들어두는 것에 불과합니다.<br /> <br /> 그렇기 때문에 위의 함수를 다음과 같이 Func<int, int>로 바꿀 수 없습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > <span style='color: blue; font-weight: bold'>Func<int, int></span> addWithDefault = (int addTo = 2) => addTo + 1; // 컴파일 경고: warning CS9099: Parameter 1 has default value '2' in lambda but '<missing>' in the target delegate type. addWithDefault(); // 컴파일 오류: error CS7036: There is no argument given that corresponds to the required parameter 'arg' of 'Func<int, int>' addWithDefault(5); </pre> <br /> 왜냐하면 Func<int, int> 델리게이트의 정의에는,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > namespace System { public delegate TResult Func<in T, out TResult>(T arg); } </pre> <br /> 기본값이 없기 때문입니다. 만약 기본값이 있는 람다 메서드를 (var 추론으로 생성하지 않고) 직접 델리게이트를 이용해 받고 싶다면 다음과 같은 델리게이트 정의를 추가해야 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > MyFunc addWithDefault = (int addTo) => addTo + 1; // 람다 자체에는 기본값이 없음! addWithDefault(); // 3 addWithDefault(5); // 6 public delegate int MyFunc(<span style='color: blue; font-weight: bold'>int addTo = 2</span>); // 기본값은 델리게이트에서! 이 코드는 C# 10 이전에도 가능! </pre> <br /> 보다시피, 기본값의 적용은 엄밀히 람다가 아닌 델리게이트 타입이었던 것입니다. 그렇기 때문에 위의 코드에서 addTo 인자에 기본값을 적용하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { MyFunc addWithDefault = (<span style='color: blue; font-weight: bold'>int addTo = 5</span>) => addTo + 1; // 컴파일 경고: warning CS9099: Parameter 1 has default value '5' in lambda but '2' in the target delegate type. Console.WriteLine(addWithDefault()); // 3 Console.WriteLine(addWithDefault(5)); // 6 } </pre> <br /> 컴파일 경고와 함께, 사용자가 지정한 5가 아닌 MyFunc 델리게이트 타입에 지정된 2가 적용되는 것을 볼 수 있습니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 특이하게도, 명시적인 델리게이트 타입을 경유한 람다의 기본값 적용은 Reflection으로 알아낼 수 없습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { MyFunc addWithDefault = (int addTo) => addTo + 1; Console.WriteLine(addWithDefault()); // 3 Console.WriteLine(addWithDefault(5)); // 6 Console.WriteLine(addWithDefault.Method.GetParameters()[0].DefaultValue); // 출력 결과: (빈 문자열) } public delegate int MyFunc(<span style='color: blue; font-weight: bold'>int addTo = 2</span>); </pre> <br /> 반면 위와 같은 상황에서 람다 측에 기본값을 적용하면, (경고도 발생하고 유효하지 않는데도 불구하고) Reflection은 람다에 정의한 기본값을 가져온다는 점입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > { MyFunc addWithDefault = (<span style='color: blue; font-weight: bold'>int addTo = 10</span>) => addTo + 1; Console.WriteLine(addWithDefault()); // 3 Console.WriteLine(addWithDefault(5)); // 6 Console.WriteLine(addWithDefault.Method.GetParameters()[0].DefaultValue); <span style='color: blue; font-weight: bold'>// 출력 결과: 10</span> } public delegate int MyFunc(<span style='color: blue; font-weight: bold'>int addTo = 2</span>); </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=2079&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> <br /> 그리고, 이미지 하나 ^^<br /> [출처: <a target='tab' href='https://nietras.com/2023/11/14/dotnet-and-csharp-versions/'>.NET and C# Versions - 8/12 Update</a><br /> <img onclick='toggle_img(this)' class='imgView' alt='dotnet-versions-export.png' src='https://nietras.com/images/2023-11-dotnet-versions/dotnet-versions-export.png' /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
9270
(왼쪽의 숫자를 입력해야 합니다.)