Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 9개 있습니다.)
닷넷: 2275. C# 13 - (1) 신규 이스케이프 시퀀스 '\e'
; https://www.sysnet.pe.kr/2/0/13673

닷넷: 2277. C# 13 - (2) 메서드 그룹의 자연 타입 개선 (메서드 추론 개선)
; https://www.sysnet.pe.kr/2/0/13681

닷넷: 2286. C# 13 - (3) Monitor를 대체할 Lock 타입
; https://www.sysnet.pe.kr/2/0/13699

닷넷: 2287. C# 13 - (4) Indexer를 이용한 개체 초기화 구문에서 System.Index 연산자 허용
; https://www.sysnet.pe.kr/2/0/13701

닷넷: 2291. C# 13 - (5) params 인자 타입으로 컬렉션 허용
; https://www.sysnet.pe.kr/2/0/13705

닷넷: 2294. C# 13 - (6) iterator 또는 비동기 메서드에서 ref와 unsafe 사용을 부분적으로 허용
; https://www.sysnet.pe.kr/2/0/13710

닷넷: 2303. C# 13 - (7) ref struct의 interface 상속 및 제네릭 제약으로 사용 가능
; https://www.sysnet.pe.kr/2/0/13752

닷넷: 2304. C# 13 - (8) 부분 메서드 정의를 속성 및 인덱서에도 확대
; https://www.sysnet.pe.kr/2/0/13754

닷넷: 2305. C# 13 - (9) 메서드 바인딩의 우선순위를 지정하는 OverloadResolutionPriority 특성 도입 (Overload resolution priority)
; https://www.sysnet.pe.kr/2/0/13755




C# 13 - (9) 메서드 바인딩의 우선순위를 지정하는 OverloadResolutionPriority 특성 도입 (Overload resolution priority)

이 기능은 현재(2024-10-10) Visual Studio 2022 Preview 버전에서 실습할 수 있습니다.




C# 언어가 지원하는 매개변수의 접근자는 대표적으로 4가지 정도가 됩니다.

  • ref
  • out
  • in
  • ref readonly

이러한 것들은 모두 메서드 시그니처에 반영되고, 그에 따라 메서드를 재정의하는 것이 가능하다 보니 경우에 따라 어떤 메서드가 호출될 것인지 예측하기가 어려워지는 경우도 있습니다. 달리 말하면, C# 컴파일러가 메서드 호출 구문을 분석할 때 실제로 어떤 메서드에 호출을 바인딩해야 하는지 판단하기가 모호할 수 있다는 것입니다.

이런 문제를, C# 13부터는 바인딩 우선순위를 개발자가 직접 OverloadResolutionPriority 특성을 추가해 컴파일러에게 힌트를 제공함으로써 해결할 수 있게 만들었습니다.

Overload resolution priority
; https://learn.microsoft.com/en-us/dotnet/csharp/whats-new/csharp-13#overload-resolution-priority

Overload Resolution Priority
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-13.0/overload-resolution-priority#overload-resolution-priority

[Proposal]: Overload Resolution Priority (VS 17.12, .NET 9) #7706
; https://github.com/dotnet/csharplang/issues/7706

OverloadResolutionPriority attribute
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/attributes/general#overloadresolutionpriority-attribute

System.Runtime.CompilerServices.OverloadResolutionPriority 
; https://learn.microsoft.com/en-us/dotnet/api/system.runtime.compilerservices.overloadresolutionpriorityattribute

실제로 이 특성을 사용하면 지난 글에 설명한 내용 중,

닷넷: 2291. C# 13 - (5) params 인자 타입으로 컬렉션 허용
; https://www.sysnet.pe.kr/2/0/13705

아래의 코드가 "error CS0121" 컴파일 에러를 발생시키는 것을 해결할 수 있습니다.

// https://www.sysnet.pe.kr/2/0/13705#cs0121

printAll([6, 7, 8, 9, 10]); // 컴파일 에러 - error CS0121: The call is ambiguous between the following methods or properties: 'Program.printAll(List<int>)' and 'Program.printAll(params ReadOnlySpan<int>)'

static void printAll(List<int> args)
{
    for (int i = 0; i < args.Count; i++)
    {
        Console.Write($"{args[i]},");
    }

    Console.WriteLine();
}

static void printAll(params ReadOnlySpan<int> args)
{
    foreach (var item in args)
    {
        Console.Write($"{item},");
    }

    Console.WriteLine();
}

지난 글에서는 위의 코드를 컴파일하기 위해 형변환 연산자를 사용했었는데요,

printAll((ReadOnlySpan<int>)[6, 7, 8, 9, 10]); // 컴파일 OK, printAll(params ReadOnlySpan<int> args) 호출로 바인딩

위와 같이 형변환을 하는 대신 OverloadResolutionPriority 특성을 이용해 바인딩 우선순위를 지정하는 것으로 해결할 수도 있습니다.

printAll([6, 7, 8, 9, 10]); // 컴파일 OK, printAll(params ReadOnlySpan<int> args) 호출로 바인딩

static void printAll(List<int> args)
{
    Console.Write("List<int> : ");
    // ...[생략]...
}

[OverloadResolutionPriority(1)]
static void printAll(params ReadOnlySpan<int> args)
{
    Console.Write("ReadOnlySpan<int> : ");
    // ...[생략]...
}

물론, "static void printAll(List<int> args)"에 [OverloadResolutionPriority(2)] 특성을 지정하면 이제는 "printAll(List<int> args)" 메서드가 더 높은 우선순위로 바인딩이 됩니다.

참고로, 번호가 클수록 우선순위가 높으며, C# 컴파일러는 OverloadResolutionPriority 특성이 없는 메서드를 "OverloadResolutionPriority(0)"으로 간주합니다.




이 기능은 라이브러리 개발자에게 특히 유용할 수 있습니다. 예를 들어, 위의 예제에서 "printAll(List<int> args)" 구현만을 포함한 v1.0 어셈블리가 있다고 가정해 보겠습니다. 그러다, 좀 더 성능이 좋은 버전인 "ReadOnlySpan<int> args" 인자를 받는 printAll을 추가해 v2.0을 내놓은 경우 기존 v1.0을 사용해 잘 컴파일을 시켰던 "printAll([6, 7, 8, 9, 10])" 코드가 이제는 컴파일 오류로 빌드 실패가 됩니다. 즉, 소스코드 수준에서의 하위 호환성이 깨진 것입니다.

반면, 라이브러리 개발자가 OverloadResolutionPriority 특성을 "ReadOnlySpan<int> args"를 받는 printAll 메서드에 부여하면, v1.0을 사용해 컴파일했던 코드가 v2.0에서도 그대로 컴파일되는 것을 보장할 수 있습니다.




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







[최초 등록일: ]
[최종 수정일: 10/10/2024]

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

비밀번호

댓글 작성자
 




1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13843정성태12/13/20244381오류 유형: 938. Docker container 내에서 빌드 시 error MSB3021: Unable to copy file "..." to "...". Access to the path '...' is denied.
13842정성태12/12/20244534디버깅 기술: 205. Windbg - KPCR, KPRCB
13841정성태12/11/20244866오류 유형: 937. error MSB4044: The "ValidateValidArchitecture" task was not given a value for the required parameter "RemoteTarget"
13840정성태12/11/20244439오류 유형: 936. msbuild - Your project file doesn't list 'win' as a "RuntimeIdentifier"
13839정성태12/11/20244859오류 유형: 936. msbuild - error CS1617: Invalid option '12.0' for /langversion. Use '/langversion:?' to list supported values.
13838정성태12/4/20244595오류 유형: 935. Windbg - Breakpoint 0's offset expression evaluation failed.
13837정성태12/3/20245058디버깅 기술: 204. Windbg - 윈도우 핸들 테이블 (3) - Windows 10 이상인 경우
13836정성태12/3/20244625디버깅 기술: 203. Windbg - x64 가상 주소를 물리 주소로 변환 (페이지 크기가 2MB인 경우)
13835정성태12/2/20245071오류 유형: 934. Azure - rm: cannot remove '...': Directory not empty
13834정성태11/29/20245287Windows: 275. C# - CUI 애플리케이션과 Console 윈도우 (Windows 10 미만의 Classic Console 모드인 경우) [1]파일 다운로드1
13833정성태11/29/20244979개발 환경 구성: 737. Azure Web App에서 Scale-out으로 늘어난 리눅스 인스턴스에 SSH 접속하는 방법
13832정성태11/27/20244911Windows: 274. Windows 7부터 도입한 conhost.exe
13831정성태11/27/20244382Linux: 111. eBPF - BPF_MAP_TYPE_PERF_EVENT_ARRAY, BPF_MAP_TYPE_RINGBUF에 대한 다양한 용어들
13830정성태11/25/20245185개발 환경 구성: 736. 파이썬 웹 앱을 Azure App Service에 배포하기
13829정성태11/25/20245162스크립트: 67. 파이썬 - Windows 버전에서 함께 설치되는 py.exe
13828정성태11/25/20244440개발 환경 구성: 735. Azure - 압축 파일을 이용한 web app 배포 시 디렉터리 구분이 안 되는 문제파일 다운로드1
13827정성태11/25/20245091Windows: 273. Windows 환경의 파일 압축 방법 (tar, Compress-Archive)
13826정성태11/21/20245320닷넷: 2313. C# - (비밀번호 등의) Console로부터 입력받을 때 문자열 출력 숨기기(echo 끄기)파일 다운로드1
13825정성태11/21/20245667Linux: 110. eBPF / bpf2go - BPF_RINGBUF_OUTPUT / BPF_MAP_TYPE_RINGBUF 사용법
13824정성태11/20/20244751Linux: 109. eBPF / bpf2go - BPF_PERF_OUTPUT / BPF_MAP_TYPE_PERF_EVENT_ARRAY 사용법
13823정성태11/20/20245290개발 환경 구성: 734. Ubuntu에 docker, kubernetes (k3s) 설치
13822정성태11/20/20245155개발 환경 구성: 733. Windbg - VirtualBox VM의 커널 디버거 연결 시 COM 포트가 없는 경우
13821정성태11/18/20245079Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식
13820정성태11/18/20245249VS.NET IDE: 195. Visual C++ - C# 프로젝트처럼 CopyToOutputDirectory 항목을 추가하는 방법
13819정성태11/15/20244490Linux: 107. eBPF - libbpf CO-RE의 CONFIG_DEBUG_INFO_BTF 빌드 여부에 대한 의존성
13818정성태11/15/20245289Windows: 272. Windows 11 24H2 - sudo 추가
1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...