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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  37  38  39  40  41  42  43  [44]  45  ...
NoWriterDateCnt.TitleFile(s)
12872정성태12/12/202116777개발 환경 구성: 615. GoLang과 Python 빌드가 모두 가능한 docker 이미지 만들기
12871정성태12/12/202115687오류 유형: 771. docker: Error response from daemon: OCI runtime create failed
12870정성태12/9/202115284개발 환경 구성: 614. 파이썬 - PyPI 패키지 만들기 (4) package_data 옵션
12869정성태12/8/202117816개발 환경 구성: 613. git clone 실행 시 fingerprint 묻는 단계를 생략하는 방법
12868정성태12/7/202116499오류 유형: 770. twine 업로드 시 "HTTPError: 400 Bad Request ..." 오류 [1]
12867정성태12/7/202116177개발 환경 구성: 612. 파이썬 - PyPI 패키지 만들기 (3) entry_points 옵션
12866정성태12/7/202123106오류 유형: 769. "docker build ..." 시 "failed to solve with frontend dockerfile.v0: failed to read dockerfile ..." 오류
12865정성태12/6/202116480개발 환경 구성: 611. 파이썬 - PyPI 패키지 만들기 (2) long_description, cmdclass 옵션
12864정성태12/6/202114151Linux: 46. WSL 환경에서 find 명령을 사용해 파일을 찾는 방법
12863정성태12/4/202116182개발 환경 구성: 610. 파이썬 - PyPI 패키지 만들기
12862정성태12/3/202113894오류 유형: 768. Golang - 빌드 시 "cmd/go: unsupported GOOS/GOARCH pair linux /amd64" 오류
12861정성태12/3/202117889개발 환경 구성: 609. 파이썬 - "Windows embeddable package"로 개발 환경 구성하는 방법 [1]
12860정성태12/1/202113925오류 유형: 767. SQL Server - 127.0.0.1로 접속하는 경우 "Access is denied"가 발생한다면?
12859정성태12/1/202122303개발 환경 구성: 608. Hyper-V 가상 머신에 Console 모드로 로그인하는 방법
12858정성태11/30/202119621개발 환경 구성: 607. 로컬의 USB 장치를 원격 머신에 제공하는 방법 - usbip-win
12857정성태11/24/202116252개발 환경 구성: 606. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법
12856정성태11/23/202118992.NET Framework: 1121. C# - 동일한 IP:Port로 바인딩 가능한 서버 소켓 [2]
12855정성태11/13/202114239개발 환경 구성: 605. Azure App Service - Kudu SSH 환경에서 FTP를 이용한 파일 전송
12854정성태11/13/202116373개발 환경 구성: 604. Azure - 윈도우 VM에서 FTP 여는 방법
12853정성태11/10/202114815오류 유형: 766. Azure App Service - JBoss 호스팅 생성 시 "This region has quota of 0 PremiumV3 instances for your subscription. Try selecting different region or SKU."
12851정성태11/1/202116522스크립트: 34. 파이썬 - MySQLdb 기본 예제 코드
12850정성태10/27/202118307오류 유형: 765. 우분투에서 pip install mysqlclient 실행 시 "OSError: mysql_config not found" 오류
12849정성태10/17/202117880스크립트: 33. JavaScript와 C#의 시간 변환 [1]
12848정성태10/17/202117017스크립트: 32. 파이썬 - sqlite3 기본 예제 코드 [1]
12847정성태10/14/202117709스크립트: 31. 파이썬 gunicorn - WORKER TIMEOUT 오류 발생
12846정성태10/7/202118759스크립트: 30. 파이썬 __debug__ 플래그 변수에 따른 코드 실행 제어
... 31  32  33  34  35  36  37  38  39  40  41  42  43  [44]  45  ...