Microsoft MVP성태의 닷넷 이야기
.NET Framework: 962. C# 9.0 - (11) 공변 반환 형식(Covariant return types) [링크 복사], [링크+제목 복사]
조회: 11461
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 15개 있습니다.)

C# 9.0 - (11) 공변 반환 형식(Covariant return types)

C# 9.0 - (1) 대상으로 형식화된 new 식(Target-typed new expressions)
; https://www.sysnet.pe.kr/2/0/12363

C# 9.0 - (2) localsinit 플래그 내보내기 무시(Suppress emitting localsinit flag)
; https://www.sysnet.pe.kr/2/0/12364

C# 9.0 - (3) 람다 메서드의 매개 변수 무시(Lambda discard parameters)
; https://www.sysnet.pe.kr/2/0/12365

C# 9.0 - (4) 원시 크기 정수(Native ints)
; https://www.sysnet.pe.kr/2/0/12366

C# 9.0 - (5) 로컬 함수에 특성 지정 가능(Attributes on local functions)
; https://www.sysnet.pe.kr/2/0/12372

C# 9.0 - (6) 함수 포인터(Function pointers)
; https://www.sysnet.pe.kr/2/0/12374

C# 9.0 - (7) 패턴 일치 개선 사항(Pattern matching enhancements)
; https://www.sysnet.pe.kr/2/0/12383

C# 9.0 - (8) 정적 익명 함수 (static anonymous functions)
; https://www.sysnet.pe.kr/2/0/12389

C# 9.0 - (9) 레코드 (Records)
; https://www.sysnet.pe.kr/2/0/12392

C# 9.0 - (10) 대상으로 형식화된 조건식(Target-typed conditional expressions)
; https://www.sysnet.pe.kr/2/0/12399

C# 9.0 - (11) 공변 반환 형식(Covariant return types)
; https://www.sysnet.pe.kr/2/0/12402

C# 9.0 - (12) foreach 루프에 대한 GetEnumerator 확장 메서드 지원(Extension GetEnumerator)
; https://www.sysnet.pe.kr/2/0/12403

C# 9.0 - (13) 모듈 이니셜라이저(Module initializers)
; https://www.sysnet.pe.kr/2/0/12404

C# 9.0 - (14) 부분 메서드에 대한 새로운 기능(New features for partial methods)
; https://www.sysnet.pe.kr/2/0/12405

C# 9.0 - (15) 최상위 문(Top-level statements)
; https://www.sysnet.pe.kr/2/0/12406

C# 9.0 - (16) 제약 조건이 없는 형식 매개변수 주석(Unconstrained type parameter annotations)
; https://www.sysnet.pe.kr/2/0/12423




(이번 실습은 - 오늘 기준 16.7.7의 Visual Studio 2019에서 지원하지 않으므로 Visual Studio Preview 버전으로 실습해야 합니다. 또한 .NET Framework 및 .NET Core 3.1 이하의 런타임에서도 동작하지 않으므로 반드시 .NET 5 환경에서 테스트해야 합니다.)

C# 언어는, 메서드의 구분을 이름과 인자의 타입으로 결정합니다. 달리 말하면, 반환 타입을 포함하지 않기 때문에 다음과 같은 식으로 메서드를 정의하면,

public class TestClass
{
    public short MyMethod(int count) { return 0; }
    public int   MyMethod(int count) { return 0; } // 컴파일 에러
}

이런 컴파일 에러 메시지를 보게 됩니다.

error CS0111: Type 'TestClass' already defines a member called 'MyMethod' with the same parameter types

얼핏 보면 이상해 보여도, 여러분들이 컴파일러 개발자라고 가정하고 다음의 코드를 어떤 메서드에 바인딩해야 할지 결정하는 순간이 되면,

MyMethod(5);  // (short MyMethod)? (int MyMethod)?

아마도 그 제약은 당연한 것이라고 자연스럽게 수긍이 될 것입니다.




이러한 제약이, 상속 관계의 반환 타입으로 오면 상황이 다소 달라집니다.

using System;

class Program
{
    static void Main(string[] args)
    {
        Product prd = new Headset().GetDevice();
        Headset hds = new Headset().GetDevice();
    }
}

public class Product
{
    public virtual Product GetDevice() { return this; }
}

public class Headset : Product
{
    // C# 8.0 이전 컴파일 오류
    // C# 9.0 + .NET 5 환경에서 컴파일 가능
    public override Headset GetDevice()
    {
        return this;
    }
}

C# 8.0까지는 위와 같이 재정의한(override) 메서드의 경우, 반환 타입이 설령 하위 타입에 해당하더라도 무조건 "Error CS0508 'Headset.GetDevice()': return type must be 'Product' to match overridden member 'Product.GetDevice()'" 컴파일 오류가 발생했지만, C# 9.0부터는 이런 제약이 풀렸습니다. 이것을 달리 말하면 "C# 메서드의 반환 타입에 공변을 허용한다."라고 표현합니다.

재미있는 것은 위의 소스 코드를 C# 9.0 + (.NET Core 3.1 이하 또는 .NET Frmaework의 모든) 버전에서 컴파일할 때 발생하는 오류 메시지입니다.

Error CS8830 'Headset.GetDevice()': Target runtime doesn't support covariant return types in overrides. Return type must be 'Product' to match overridden member 'Product.GetDevice()'

즉, 공변 반환 타입의 제약은 C# 언어만의 문제가 아니라 CLR 런타임 쪽에서도 허용을 하지 않았던 것입니다.

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




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/22/2020]

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

비밀번호

댓글 작성자
 



2020-11-09 10시46분
C# 9.0: 공변 리턴 타입 (Covariant return type)
http://www.csharpstudy.com/latest/CS9-covariant-return.aspx
정성태

1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13429정성태10/27/20232923닷넷: 2152. Win32 Interop - C/C++ DLL로부터 이중 포인터 버퍼를 C#으로 받는 예제파일 다운로드1
13428정성태10/25/20232963닷넷: 2151. C# 12 - ref readonly 매개변수
13427정성태10/18/20233154닷넷: 2150. C# 12 - 정적 문맥에서 인스턴스 멤버에 대한 nameof 접근 허용(Allow nameof to always access instance members from static context)
13426정성태10/13/20233323스크립트: 59. 파이썬 - 비동기 호출 함수(run_until_complete, run_in_executor, create_task, run_in_threadpool)
13425정성태10/11/20233140닷넷: 2149. C# - PLinq의 Partitioner<T>를 이용한 사용자 정의 분할파일 다운로드1
13423정성태10/6/20233115스크립트: 58. 파이썬 - async/await 기본 사용법
13422정성태10/5/20233269닷넷: 2148. C# - async 유무에 따른 awaitable 메서드의 병렬 및 예외 처리
13421정성태10/4/20233313닷넷: 2147. C# - 비동기 메서드의 async 예약어 유무에 따른 차이
13420정성태9/26/20235461스크립트: 57. 파이썬 - UnboundLocalError: cannot access local variable '...' where it is not associated with a value
13419정성태9/25/20233148스크립트: 56. 파이썬 - RuntimeError: dictionary changed size during iteration
13418정성태9/25/20233840닷넷: 2146. C# - ConcurrentDictionary 자료 구조의 동기화 방식
13417정성태9/19/20233396닷넷: 2145. C# - 제네릭의 형식 매개변수에 속한 (매개변수를 가진) 생성자를 호출하는 방법
13416정성태9/19/20233186오류 유형: 877. redis-py - MISCONF Redis is configured to save RDB snapshots, ...
13415정성태9/18/20233677닷넷: 2144. C# 12 - 컬렉션 식(Collection Expressions)
13414정성태9/16/20233446디버깅 기술: 193. Windbg - ThreadStatic 필드 값을 조사하는 방법
13413정성태9/14/20233641닷넷: 2143. C# - 시스템 Time Zone 변경 시 이벤트 알림을 받는 방법
13412정성태9/14/20236918닷넷: 2142. C# 12 - 인라인 배열(Inline Arrays) [1]
13411정성태9/12/20233413Windows: 252. 권한 상승 전/후 따로 관리되는 공유 네트워크 드라이브 정보
13410정성태9/11/20234927닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성) [1]
13409정성태9/8/20233784닷넷: 2140. C# - Win32 API를 이용한 모니터 전원 끄기
13408정성태9/5/20233762Windows: 251. 임의로 만든 EXE 파일을 포함한 ZIP 파일의 압축을 해제할 때 Windows Defender에 의해 삭제되는 경우
13407정성태9/4/20233515닷넷: 2139. C# - ParallelEnumerable을 이용한 IEnumerable에 대한 병렬 처리
13406정성태9/4/20233476VS.NET IDE: 186. Visual Studio Community 버전의 라이선스
13405정성태9/3/20233919닷넷: 2138. C# - async 메서드 호출 원칙
13404정성태8/29/20233427오류 유형: 876. Windows - 키보드의 등호(=, Equals sign) 키가 눌리지 않는 경우
13403정성태8/21/20233264오류 유형: 875. The following signatures couldn't be verified because the public key is not available: NO_PUBKEY EB3E94ADBE1229CF
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...