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

비밀번호

댓글 작성자
 




... 91  92  93  94  95  96  97  98  99  100  101  [102]  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11383정성태12/4/201723392디버깅 기술: 110. 비동기 코드 실행 중 예외로 인한 ASP.NET 프로세스 비정상 종료 현상 [1]
11382정성태12/4/201721924오류 유형: 436. System.Data.SqlClient.SqlException (0x80131904): Connection Timeout Expired 예외 발생 시 "[Pre-Login] initialization=48; handshake=1944;" 값의 의미
11381정성태11/30/201718414.NET Framework: 702. 한글이 포함된 바이트 배열을 나눈 경우 한글이 깨지지 않도록 다시 조합하는 방법(두 번째 이야기)파일 다운로드1
11380정성태11/30/201718440디버깅 기술: 109. windbg - (x64에서의 인자 값 추적을 이용한) Thread.Abort 시 대상이 되는 스레드를 식별하는 방법
11379정성태11/30/201719136오류 유형: 435. System.Web.HttpException - Session state has created a session id, but cannot save it because the response was already flushed by the application.
11378정성태11/29/201720615.NET Framework: 701. 한글이 포함된 바이트 배열을 나눈 경우 한글이 깨지지 않도록 다시 조합하는 방법 [1]파일 다운로드1
11377정성태11/29/201719876.NET Framework: 700. CommonOpenFileDialog 사용 시 사용자가 선택한 파일 목록을 구하는 방법 [3]파일 다운로드1
11376정성태11/28/201724269VS.NET IDE: 123. Visual Studio 편집기의 \r\n (crlf) 개행을 \n으로 폴더 단위로 설정하는 방법
11375정성태11/28/201719064오류 유형: 434. Visual Studio로 ASP.NET 디버깅 중 System.Web.HttpException - Could not load type 오류
11374정성태11/27/201724159사물인터넷: 14. 라즈베리 파이 - (윈도우의 NT 서비스처럼) 부팅 시 시작하는 프로그램 설정 [1]
11373정성태11/27/201723146오류 유형: 433. Raspberry Pi/Windows 다중 플랫폼 지원 컴파일 관련 오류 기록
11372정성태11/25/201726130사물인터넷: 13. 윈도우즈 사용자를 위한 라즈베리 파이 제로 W 모델을 설정하는 방법 [4]
11371정성태11/25/201719805오류 유형: 432. Hyper-V 가상 스위치 생성 시 Failed to connect Ethernet switch port 0x80070002 오류 발생
11370정성태11/25/201719816오류 유형: 431. Hyper-V의 Virtual Switch 생성 시 "External network" 목록에 특정 네트워크 어댑터 항목이 없는 경우
11369정성태11/25/201721777사물인터넷: 12. Raspberry Pi Zero(OTG)를 다른 컴퓨터에 연결해 가상 키보드 및 마우스로 쓰는 방법 (절대 좌표, 상대 좌표, 휠) [1]
11368정성태11/25/201727415.NET Framework: 699. UDP 브로드캐스트 주소 255.255.255.255와 192.168.0.255의 차이점과 이를 고려한 C# UDP 서버/클라이언트 예제 [2]파일 다운로드1
11367정성태11/25/201727486개발 환경 구성: 337. 윈도우 운영체제의 route 명령어 사용법
11366정성태11/25/201719134오류 유형: 430. 이벤트 로그 - Cryptographic Services failed while processing the OnIdentity() call in the System Writer Object.
11365정성태11/25/201721378오류 유형: 429. 이벤트 로그 - User Policy could not be updated successfully
11364정성태11/24/201723326사물인터넷: 11. Raspberry Pi Zero(OTG)를 다른 컴퓨터에 연결해 가상 마우스로 쓰는 방법 (절대 좌표) [2]
11363정성태11/23/201723337사물인터넷: 10. Raspberry Pi Zero(OTG)를 다른 컴퓨터에 연결해 가상 마우스 + 키보드로 쓰는 방법 (두 번째 이야기)
11362정성태11/22/201719743오류 유형: 428. 윈도우 업데이트 KB4048953 - 0x800705b4 [2]
11361정성태11/22/201722537오류 유형: 427. 이벤트 로그 - Filter Manager failed to attach to volume '\Device\HarddiskVolume??' 0xC03A001C
11360정성태11/22/201722384오류 유형: 426. 이벤트 로그 - The kernel power manager has initiated a shutdown transition.
11359정성태11/16/201721891오류 유형: 425. 윈도우 10 Version 1709 (OS Build 16299.64) 업그레이드 시 발생한 문제 2가지
11358정성태11/15/201726685사물인터넷: 9. Visual Studio 2017에서 Raspberry Pi C++ 응용 프로그램 제작 [1]
... 91  92  93  94  95  96  97  98  99  100  101  [102]  103  104  105  ...