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

C# - 제네릭 타입/메서드에서 사용 시 경우에 따라 CS8377 컴파일 에러

예를 들어 다음의 코드를 컴파일하면,

using System.Buffers;

class Program
{
    static void Main(string[] args) { }

    static void TestMethod<T>(T item)
    {
        SequenceReader<T> reader = new SequenceReader<T>();
    }
}

이런 컴파일 오류가 발생합니다.

Error CS8377 The type 'T' must be a non-nullable value type, along with all fields at any level of nesting, in order to use it as parameter 'T' in the generic type or method 'SequenceReader<T>'

그러고 보니 CS8377 오류에 대해서는 전에도 한 번 다룬 적이 있습니다.

C# 7.3 - unmanaged(blittable) 제네릭 제약
; https://www.sysnet.pe.kr/2/0/11558

{
    System.String txt = "TEST";

    // 컴파일 에러 - Error CS8377 The type 'string' must be a non-nullable value type, along with all fields at any level of nesting, in order to use it as parameter 'T' in the generic type or method 'CallGenerics.AcceptBlittable<T>(T)'
    CallGenerics.AcceptBlittable(txt); 
}

class CallGenerics
{
    public static void AcceptBlittable<T>(T item) where T : unmanaged
    {
    }
}

단지, 이번에는 제네릭 메서드 내에 형식 매개변수로 연결한 것이어서 약간 더 혼란을 주었던 것뿐입니다. 어쨌든 중요한 것은 non-nullable 타입이어야 한다고 오류 메시지에서 알려 주고 있으므로 결국 unmanaged 제약을 걸어주면 됩니다.

static void TestMethod<T>(T item) where T : unmanaged
{
    SequenceReader<T> reader = new SequenceReader<T>();
}




참고로, 위의 SequenceReader 사용 예제는 저렇게 해도 다음과 같은 컴파일 오류가 발생합니다.

Error CS0314 The type 'T' cannot be used as type parameter 'T' in the generic type or method 'SequenceReader<T>'. There is no boxing conversion or type parameter conversion from 'T' to 'System.IEquatable<T>'.

그래도 이번에는 CS8377 오류보다 직접적인 원인을 제시하고 있는데요, 따라서 System.IEquatable<T> 제약을 추가하면 됩니다.

static void TestMethod<T>(T item) where T : unmanaged, IEquatable<T>
{
    SequenceReader<T> reader = new SequenceReader<T>();
}

실제로 역어셈블을 통해 SequenceReader 타입을 보면 간단하게 모든 제약을 구할 수 있습니다.

// 문서에서는 "public struct SequenceReader<T> where T : struct"라고 나옵니다.

public ref struct SequenceReader<T> where T : unmanaged, IEquatable<T>
{
    // ...[생략]...
}

그러니까, 결국 제네릭 타입/메서드를 구현하는 경우 내부에 품은 제네릭 타입/메서드의 제약을 그대로 이어받아야 하는 것입니다.




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







[최초 등록일: ]
[최종 수정일: 1/8/2021]

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)
13198정성태12/18/202214041.NET Framework: 2080. C# - Microsoft.XmlSerializer.Generator 처리 없이 XmlSerializer 생성자를 예외 없이 사용하고 싶다면?파일 다운로드1
13197정성태12/17/202213557.NET Framework: 2079. .NET Core/5+ 환경에서 XmlSerializer 사용 시 System.IO.FileNotFoundException 예외 발생하는 경우파일 다운로드1
13196정성태12/16/202214269.NET Framework: 2078. .NET Core/5+를 위한 SGen(Microsoft.XmlSerializer.Generator) 사용법
13195정성태12/15/202214509개발 환경 구성: 655. docker - bridge 네트워크 모드에서 컨테이너 간 통신 시 --link 옵션 권장 이유
13194정성태12/14/202214321오류 유형: 833. warning C4747: Calling managed 'DllMain': Managed code may not be run under loader lock파일 다운로드1
13193정성태12/14/202214845오류 유형: 832. error C7681: two-phase name lookup is not supported for C++/CLI or C++/CX; use /Zc:twoPhase-
13192정성태12/13/202214832Linux: 55. 리눅스 - bash shell에서 실수 연산
13191정성태12/11/202216408.NET Framework: 2077. C# - 직접 만들어 보는 SynchronizationContext파일 다운로드1
13190정성태12/9/202218431.NET Framework: 2076. C# - SynchronizationContext 기본 사용법파일 다운로드1
13189정성태12/9/202217844오류 유형: 831. Visual Studio - Windows Forms 디자이너의 도구 상자에 컨트롤이 보이지 않는 문제
13188정성태12/9/202217284.NET Framework: 2075. C# - 직접 만들어 보는 TaskScheduler 실습 (SingleThreadTaskScheduler) [1]파일 다운로드1
13187정성태12/8/202216754개발 환경 구성: 654. openssl - CA로부터 인증받은 새로운 인증서를 생성하는 방법 (2)
13186정성태12/6/202214645오류 유형: 831. The framework 'Microsoft.AspNetCore.App', version '...' was not found.
13185정성태12/6/202215599개발 환경 구성: 653. Windows 환경에서의 Hello World x64 어셈블리 예제 (NASM 버전)
13184정성태12/5/202213477개발 환경 구성: 652. ml64.exe와 link.exe x64 실행 환경 구성 [1]
13183정성태12/4/202213341오류 유형: 830. MASM + CRT 함수를 사용하는 경우 발생하는 컴파일 오류 정리 [1]
13182정성태12/4/202215206Windows: 217. Windows 환경에서의 Hello World x64 어셈블리 예제 (MASM 버전)
13181정성태12/3/202213945Linux: 54. 리눅스/WSL - hello world 어셈블리 코드 x86/x64 (nasm)
13180정성태12/2/202215197.NET Framework: 2074. C# - 스택 메모리에 대한 여유 공간 확인하는 방법파일 다운로드1
13179정성태12/2/202213752Windows: 216. Windows 11 - 22H2 업데이트 이후 Terminal 대신 cmd 창이 뜨는 경우
13178정성태12/1/202214863Windows: 215. Win32 API 금지된 함수 - IsBadXxxPtr 유의 함수들이 안전하지 않은 이유파일 다운로드1
13177정성태11/30/202215439오류 유형: 829. uwsgi 설치 시 fatal error: Python.h: No such file or directory
13176정성태11/29/202212219오류 유형: 828. gunicorn - ModuleNotFoundError: No module named 'flask'
13175정성태11/29/202216882오류 유형: 827. Python - ImportError: cannot import name 'html5lib' from 'pip._vendor'
13174정성태11/28/202213350.NET Framework: 2073. C# - VMMap처럼 스택 메모리의 reserve/guard/commit 상태 출력파일 다운로드1
13173정성태11/27/202214382.NET Framework: 2072. 닷넷 응용 프로그램의 스레드 스택 크기 변경
... [31]  32  33  34  35  36  37  38  39  40  41  42  43  44  45  ...