Microsoft MVP성태의 닷넷 이야기
닷넷: 2112. C# 12 - 기본 람다 매개 변수 [링크 복사], [링크+제목 복사],
조회: 11909
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 9개 있습니다.)
닷넷: 2112. C# 12 - 기본 람다 매개 변수
; https://www.sysnet.pe.kr/2/0/13338

닷넷: 2113. C# 12 - 기본 생성자(Primary Constructors)
; https://www.sysnet.pe.kr/2/0/13339

닷넷: 2114. C# 12 - 모든 형식의 별칭(Using aliases for any type)
; https://www.sysnet.pe.kr/2/0/13341

닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성)
; https://www.sysnet.pe.kr/2/0/13410

닷넷: 2142. C# 12 - 인라인 배열(Inline Arrays)
; https://www.sysnet.pe.kr/2/0/13412

닷넷: 2144. C# 12 - 컬렉션 식(Collection Expressions)
; https://www.sysnet.pe.kr/2/0/13415

닷넷: 2150. C# 12 - 정적 문맥에서 인스턴스 멤버에 대한 nameof 접근 허용(Allow nameof to always access instance members from static context)
; https://www.sysnet.pe.kr/2/0/13427

닷넷: 2151. C# 12 - ref readonly 매개변수
; https://www.sysnet.pe.kr/2/0/13428

닷넷: 2160. C# 12 - Experimental 특성 지원
; https://www.sysnet.pe.kr/2/0/13444




C# 12 - 기본 람다 매개 변수

("람다 식의 선택적 매개 변수"라고 예전에 제목을 지었는데 공식 문서에 "기본 람다 매개 변수"로 나와서 그에 따라 이름을 바꿉니다.)




C# Next, 즉 C# 12에 포함될 기능에 대한 진행 사항이 아래의 문서를 통해 공개돼 있습니다.

C# Next
; https://github.com/dotnet/roslyn/blob/main/docs/Language%20Feature%20Status.md#c-next

.NET Conf 2023 - "What's New in C# 12"
; https://youtu.be/xEFO1sQ2bUc?t=16316

12개의 개선이 추가되는데 오늘은 그 첫 번째로 "Lambda default parameters"를 소개합니다. ^^ 이것을 먼저 선택한 이유는, 해당 기능은 현재 버전의 Visual Studio 2022 17.5.5 버전에서 테스트할 수 있기 때문입니다. (다른 기능들은 Preview 버전의 Visual Studio 2022를 요구합니다.)

따라서 이 글의 실습을 하려면 프로젝트 생성 후 LangVersion만 preview로 바꾸면 됩니다.

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>net7.0</TargetFramework>
    <ImplicitUsings>enable</ImplicitUsings>
    <Nullable>enable</Nullable>
    <LangVersion>preview</LangVersion>
  </PropertyGroup>

</Project>




C#에서 발전 중인 Lambda의 역사는 다음과 같이 정리할 수 있습니다

  • C# 3.0: 처음 람다 지원
  • C# 6.0: 람다 표현식을 이용해 일반 메서드, 속성과 인덱서의 get 접근자에 대한 정의
  • C# 7.0: C# 6.0에 구현된 범위를 확장 - 생성자, 종료자, 이벤트의 add/remove, 속성과 인덱서의 set 접근자
  • C# 9.0: 매개변수 무시, static 지원
  • C# 10: 특성 허용, 반환 타입 지정, var 추론

뭐랄까, 일반 메서드에서 허용된 기능들이 람다에서도 지원되고 있는 건데요, 사실 람다뿐만 아니라 익명 메서드, 로컬 함수까지 전방위적으로 지속적인 개선이 이뤄지고 있습니다.

그리고 C# 12에는 람다에 대해 (일반 메서드에나 가능했던) 매개변수의 기본값까지 설정하는 것이 가능해졌습니다.

var addWithDefault = (int addTo = 2) => addTo + 1;
addWithDefault(); // 3
addWithDefault(5); // 6

여기서 재미있는 것은, 람다 자체를 받는 것은 결국 delegate 타입이기 때문에 사실상 기본값에 대한 설정은 람다가 아닌 delegate에 있다는 점입니다. 단지, 위의 코드는 var가 그것을 추론해 대응하는 delegate 타입을 C# 컴파일러가 대신 만들어두는 것에 불과합니다.

그렇기 때문에 위의 함수를 다음과 같이 Func<int, int>로 바꿀 수 없습니다.

Func<int, int> 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);

왜냐하면 Func<int, int> 델리게이트의 정의에는,

namespace System
{
    public delegate TResult Func<in T, out TResult>(T arg);
}

기본값이 없기 때문입니다. 만약 기본값이 있는 람다 메서드를 (var 추론으로 생성하지 않고) 직접 델리게이트를 이용해 받고 싶다면 다음과 같은 델리게이트 정의를 추가해야 합니다.

MyFunc addWithDefault = (int addTo) => addTo + 1; // 람다 자체에는 기본값이 없음!
addWithDefault(); // 3
addWithDefault(5); // 6

public delegate int MyFunc(int addTo = 2); // 기본값은 델리게이트에서! 이 코드는 C# 10 이전에도 가능!

보다시피, 기본값의 적용은 엄밀히 람다가 아닌 델리게이트 타입이었던 것입니다. 그렇기 때문에 위의 코드에서 addTo 인자에 기본값을 적용하면,

{
    MyFunc addWithDefault = (int addTo = 5) => 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
}

컴파일 경고와 함께, 사용자가 지정한 5가 아닌 MyFunc 델리게이트 타입에 지정된 2가 적용되는 것을 볼 수 있습니다.




특이하게도, 명시적인 델리게이트 타입을 경유한 람다의 기본값 적용은 Reflection으로 알아낼 수 없습니다.

{
    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(int addTo = 2);

반면 위와 같은 상황에서 람다 측에 기본값을 적용하면, (경고도 발생하고 유효하지 않는데도 불구하고) Reflection은 람다에 정의한 기본값을 가져온다는 점입니다.

{
    MyFunc addWithDefault = (int addTo = 10) => addTo + 1;
    Console.WriteLine(addWithDefault()); // 3
    Console.WriteLine(addWithDefault(5)); // 6

    Console.WriteLine(addWithDefault.Method.GetParameters()[0].DefaultValue); // 출력 결과: 10
}

public delegate int MyFunc(int addTo = 2);

(첨부 파일은 이 글의 예제 코드를 포함합니다.)

그리고, 이미지 하나 ^^
[출처: .NET and C# Versions - 8/12 Update
dotnet-versions-export.png




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/20/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
12012정성태8/28/201926775.NET Framework: 859. C# - HttpListener를 이용한 HTTPS 통신 방법
12011정성태8/27/201926327사물인터넷: 57. C# - Rapsberry Pi Zero W와 PC 간 Bluetooth 통신 예제 코드파일 다운로드1
12010정성태8/27/201919259VS.NET IDE: 138. VSIX - DTE.ItemOperations.NewFile 메서드에서 템플릿 이름을 다국어로 설정하는 방법
12009정성태8/26/201920068.NET Framework: 858. C#/Windows - Clipboard(Ctrl+C, Ctrl+V)가 동작하지 않는다면?파일 다운로드1
12008정성태8/26/201919776.NET Framework: 857. UWP 앱에서 SQL Server 데이터베이스 연결 방법
12007정성태8/24/201918386.NET Framework: 856. .NET Framework 버전을 올렸을 때 오류가 발생할 수 있는 상황
12006정성태8/23/201921822디버깅 기술: 129. guidgen - Encountered an improper argument. 오류 해결 방법 (및 windbg 분석) [1]
12005정성태8/13/201919408.NET Framework: 855. 닷넷 (및 VM 계열 언어) 코드의 성능 측정 시 주의할 점 [2]파일 다운로드1
12004정성태8/12/201927713.NET Framework: 854. C# - 32feet.NET을 이용한 PC 간 Bluetooth 통신 예제 코드 [14]
12003정성태8/12/201919864오류 유형: 564. Visual C++ 컴파일 오류 - fatal error C1090: PDB API call failed, error code '3'
12002정성태8/12/201919217.NET Framework: 853. Excel Sheet를 WinForm에서 사용하는 방법 - 두 번째 이야기 [5]
12001정성태8/10/201924447.NET Framework: 852. WPF/WinForm에서 UWP의 기능을 이용해 Bluetooth 기기와 Pairing하는 방법 [1]
12000정성태8/9/201923853.NET Framework: 851. WinForm/WPF에서 Console 창을 띄워 출력하는 방법파일 다운로드1
11999정성태8/1/201918020오류 유형: 563. C# - .NET Core 2.0 이하의 Unix Domain Socket 사용 시 System.IndexOutOfRangeException 오류
11998정성태7/30/201920223오류 유형: 562. .NET Remoting에서 서비스 호출 시 SYN_SENT로 남는 현상파일 다운로드1
11997정성태7/30/201920473.NET Framework: 850. C# - Excel(을 비롯해 Office 제품군) COM 객체를 제어 후 Excel.exe 프로세스가 남아 있는 문제 [2]파일 다운로드1
11996정성태7/25/201923460.NET Framework: 849. C# - Socket의 TIME_WAIT 상태를 없애는 방법파일 다운로드1
11995정성태7/23/201927285.NET Framework: 848. C# - smtp.daum.net 서비스(Implicit SSL)를 이용해 메일 보내는 방법 [2]
11994정성태7/22/201921875개발 환경 구성: 454. Azure 가상 머신(VM)에서 SMTP 메일 전송하는 방법파일 다운로드1
11993정성태7/22/201916571오류 유형: 561. Dism.exe 수행 시 "Error: 2 - The system cannot find the file specified." 오류 발생
11992정성태7/22/201918724오류 유형: 560. 서비스 관리자 실행 시 "Windows was unable to open service control manager database on [...]. Error 5: Access is denied." 오류 발생
11991정성태7/18/201915749디버깅 기술: 128. windbg - x64 환경에서 닷넷 예외가 발생한 경우 인자를 확인할 수 없었던 사례
11990정성태7/18/201917998오류 유형: 559. Settings / Update & Security 화면 진입 시 프로그램 종료
11989정성태7/18/201916847Windows: 162. Windows Server 2019 빌드 17763부터 Alt + F4 입력시 곧바로 로그아웃하는 현상
11988정성태7/18/201919350개발 환경 구성: 453. 마이크로소프트가 지정한 모든 Root 인증서를 설치하는 방법
11987정성태7/17/201925334오류 유형: 558. 윈도우 - KMODE_EXCEPTION_NOT_HANDLED 블루스크린(BSOD) 문제 [1]
... 76  [77]  78  79  80  81  82  83  84  85  86  87  88  89  90  ...