Microsoft MVP성태의 닷넷 이야기
닷넷: 2114. C# 12 - 모든 형식의 별칭(Using aliases for any type) [링크 복사], [링크+제목 복사],
조회: 11967
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

(시리즈 글이 9개 있습니다.)
닷넷: 2112. C# 12 - 기본 람다 매개 변수
; https://www.sysnet.pe.kr/2/0/13338

닷넷: 2113. C# 12 - 기본 생성자(Primary Constructors)
; https://www.sysnet.pe.kr/2/0/13339

닷넷: 2114. C# 12 - 모든 형식의 별칭(Using aliases for any type)
; https://www.sysnet.pe.kr/2/0/13341

닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성)
; https://www.sysnet.pe.kr/2/0/13410

닷넷: 2142. C# 12 - 인라인 배열(Inline Arrays)
; https://www.sysnet.pe.kr/2/0/13412

닷넷: 2144. C# 12 - 컬렉션 식(Collection Expressions)
; https://www.sysnet.pe.kr/2/0/13415

닷넷: 2150. C# 12 - 정적 문맥에서 인스턴스 멤버에 대한 nameof 접근 허용(Allow nameof to always access instance members from static context)
; https://www.sysnet.pe.kr/2/0/13427

닷넷: 2151. C# 12 - ref readonly 매개변수
; https://www.sysnet.pe.kr/2/0/13428

닷넷: 2160. C# 12 - Experimental 특성 지원
; https://www.sysnet.pe.kr/2/0/13444




C# 12 - 모든 형식의 별칭(Using aliases for any type)

공식 문서 (Using aliases for any type)
; https://learn.microsoft.com/ko-kr/dotnet/csharp/whats-new/csharp-12#alias-any-type




기존에도 using을 이용하면 네임스페이스 및 타입에 대해 별칭을 지정할 수 있었습니다.

using My = Universe; // 네임스페이스의 별칭
using Cio = System.Console; // 타입의 별칭
using static System.Diagnostics.Trace; // 편의상 정적 타입명 생략도 지원

internal class Program
{
    private static void Main(string[] args)
    {
        My.Galaxy instance = new My.Galaxy { Name = "Milky way galaxy" };
        Cio.WriteLine(instance.Name);

        WriteLine("TEST"); // System.Diagnostics.Trace
    }
}

namespace Universe 
{
    public class Galaxy
    {
        public string? Name { get; set; }
    }
}

이름 충돌을 방지하기 위해 제공하던 별칭을 이제 C# 12부터는, 이름이 없는 타입(unnamed type), 포인터 타입, 널 가능한 타입에 대해서도 별칭을 지정하는 것이 가능해졌습니다. 즉, 어떤 타입이든 별칭을 만들 수 있게 된 것입니다.




우선 이름이 없는 타입에 대한 별칭을 볼까요? ^^

using Person = (int Age, string Name);

internal class Program
{
    private static void Main(string[] args)
    {
        Person person = new Person { Age = 1, Name = "Anders" };
        Console.WriteLine(person.Age);
    }
}

어찌 보면, 이름이 없는 타입에 별칭을 부여함으로써 이름을 가진 타입이 된 것입니다. 여기서 재미있는 것은, 별칭으로 지정한 타입은 다음과 같이 "new" 없이 초기화하는 것도 가능하다는 점입니다.

{
    Person person = (1, "Anders");
    Console.WriteLine(person.Age);
}

어떻게 이런 표현이 가능한 것일까요? ^^ 왜냐하면, 사실상 별칭으로 지정한 타입은 내부적으로 튜플로 처리하기 때문입니다. 즉, 위와 같은 경우 역어셈블 시 다음과 같은 식으로 치환이 되는데요,

using System;
using System.Runtime.CompilerServices;

internal class Program
{
    [NullableContext(1)]
    private static void Main(string[] args)
    {
        // Person person = (1, "Anders"); 코드가 아래와 같이 변환됨
        ValueTuple<int, string> valueTuple = new ValueTuple<int, string>(1, "Anders");

        // Person person = new Person { Age = 1, Name = "Anders" }; 코드가 아래와 같이 변환됨
        ValueTuple<int, string> valueTuple2 = new ValueTuple<int, string>
        {
            Item1 = 1,
            Item2 = "Anders"
        };
    }
}

튜플에 대한 C#의 단축 표현을 고려해 보면,

ValueTuple<int, string> t = (42, "Anders");

unnamed 타입에 대한 new 없는 형식이 이해가 됩니다. 따라서 만약 unnamed 타입을 별칭으로 받을 때 매개변수 이름을 지정하지 않는다면,

using Person = (int, string);

개별 요소에 대한 접근도 (튜플이므로) Item1, Item2, ... 식으로 접근하게 됩니다.

{
    Person person = (1, "Anders");
    Console.WriteLine(person.Item1); // Item1 == 첫 번째 int 요소 접근
}




그다음 포인터 타입에 대해서도 별칭이 가능합니다.

using unsafe BytePtr = byte*;

internal class Program
{
    private unsafe static void Main(string[] args)
    {
        byte[] buffer = new byte[] { 1, 2, 3 };
        fixed(BytePtr ptr = buffer)
        {
            Console.WriteLine(*ptr); // 출력 결과: 1
        }
    }
}

포인터를 다루므로 자연스럽게 "unsafe"를 지정해야 하는 차이 정도만 있을 뿐 말 그대로 별칭으로 동작합니다.

널 가능한 타입(NRT: Nullable Reference Type)에 대해서도 마저 알아볼까요? ^^ 이에 대해서는 한 가지 제약을 두고 있는데요, 바로 "참조형"의 NRT에 대해서는 별칭을 둘 수 없다는 것입니다.

즉, 다음의 코드는,

using Utf16String = System.String?; // 컴파일 에러: error CS9132: Using alias cannot be a nullable reference type.

허용하지 않으므로 System.String 그대로 별칭을 정의해야 합니다.

using Utf16String = string; // C# 12에서 가능

using Utf16String = System.String; // C# 11 이하에서도 가능

얼핏 이런 제약이 이상해 보이지만, 이것이 기존의 NRT 표현을 동일한 정책으로 적용할 수 있기 때문에,

// <Nullable>enable</Nullable> 문맥에서,
Utf16String t1 = ""; // 경고 없이 컴파일
Utf16String? t2 = null; // 경고 없이 컴파일

Utf16String t3 = null; // 경고: warning CS8600: Converting null literal or possible null value to non-nullable type.

위의 코드를 보면 일면 이해가 됩니다. 참조 형식의 NRT에 대해서만 제약이 있기 때문에 그 외의 별칭은 모두 유효합니다.

// 결국 List에 대한 별칭이고, NRT 타입이 아니므로 가능
using X1 = System.Collections.Generic.List<string?>;

// 참조 타입의 NRT가 아니므로 가능
using X2 = int?;

한 가지 재미있는 것은, using 별칭에서는 네임스페이스 선언에 독립적인 듯해서 다음과 같이 정의하면,

using System.Collections.Generic;
using X = List<string>; // error CS0246: The type or namespace name 'List<>' could not be found (are you missing a using directive or an assembly reference?)

타입을 찾을 수 없다는 컴파일 오류가 발생합니다. 현재 Preview 버전 단계라서 그런 것인지 지켜봐야 할 것 같습니다. ^^




참고로, C# 3.0부터 구현된 익명(anonymouse) 타입과 이름이 없는 타입(unnamed type)은 다릅니다. 가령 아래와 같이 익명 타입을 사용하면,

var p = new { Age = 16, Name = "Anders" }; // 익명 타입
Console.WriteLine(p.Name); // 출력 결과: "Anders"

내부적으로 C# 컴파일러가 "<>f__AnonymousType0`2"라는 식의 이름으로 된 클래스를 만들어 대응하게 됩니다.

public class <>f__AnonymousType0`2
{
    public <>f__AnonymousType0`2(int age, string name)
    {
    }

    // Age, Name 속성
    // Equals, GetHashCode, ToString 재정의
}

반면 별칭으로 지정한 이름 없는 타입은 튜플로 처리가 되는 것과는 차이가 있습니다. 실제로, 익명 타입의 정의를 별칭에 사용할 수는 없습니다.

using Student = { int, string }; // 컴파일 에러

따라서 이름 없는 타입에 대한 별칭이라고는 하지만, 엄밀히 Tuple에 대한 별칭을 지원한다고 보면 되고 그런 의미에서 봤을 때 사실 기존 C# 11 이하에서도 다음과 같은 식의 별칭을 두는 것은 유효했기 때문에,

using Person = System.ValueTuple<int, string>;

Person p1 = (42, "Anders");
Console.WriteLine(p1); // 출력 결과: (42, Anders)

C# 12에서는 좀 더 간결한 지원을 추가했다고 봐도 아주 다른 이야기는 아닙니다. ^^




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







[최초 등록일: ]
[최종 수정일: 12/13/2023]

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

비밀번호

댓글 작성자
 




... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...
NoWriterDateCnt.TitleFile(s)
1303정성태6/26/201227403개발 환경 구성: 152. sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
1302정성태6/25/201229445개발 환경 구성: 151. Azure 웹 사이트에 사용자 도메인 네임 연결하는 방법
1301정성태6/20/201225765오류 유형: 156. KB2667402 윈도우 업데이트 실패 및 마이크로소프트 Answers 웹 사이트 대응
1300정성태6/20/201231776.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 [1]파일 다운로드2
1299정성태6/18/201232892제니퍼 .NET: 21. 제니퍼 닷넷 - Ninject DI 프레임워크의 성능 분석 [2]파일 다운로드2
1298정성태6/14/201234407VS.NET IDE: 72. Visual Studio에서 pfx 파일로 서명한 경우, 암호는 어디에 저장될까? [2]
1297정성태6/12/201231055VC++: 63. 다른 프로세스에 환경 변수 설정하는 방법파일 다운로드1
1296정성태6/5/201227695.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 [4]파일 다운로드1
1295정성태6/5/201225083.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기파일 다운로드1
1294정성태5/27/201248531.NET Framework: 326. 유니코드와 한글 - 유니코드와 닷넷을 이용한 한글 처리 [7]파일 다운로드2
1293정성태5/24/201229774.NET Framework: 325. System.Drawing.Bitmap 데이터를 Parallel.For로 처리하는 방법 [2]파일 다운로드1
1292정성태5/24/201223754.NET Framework: 324. First-chance exception에 대해 조건에 따라 디버거가 멈추게 할 수는 없을까? [1]파일 다운로드1
1291정성태5/23/201230279VC++: 62. 배열 초기화를 위한 기계어 코드 확인 [2]
1290정성태5/18/201235081.NET Framework: 323. 관리자 권한이 필요한 작업을 COM+에 대행 [7]파일 다운로드1
1289정성태5/17/201239239.NET Framework: 322. regsvcs.exe로 어셈블리 등록 시 시스템 변경 사항 [5]파일 다운로드2
1288정성태5/17/201226465.NET Framework: 321. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (3) - Type Library파일 다운로드1
1287정성태5/17/201229299.NET Framework: 320. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (2) - .NET 4.0 + .NET 2.0 [2]
1286정성태5/17/201238228.NET Framework: 319. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (1) - .NET 2.0 + x86/x64/AnyCPU [5]
1285정성태5/16/201233265.NET Framework: 318. gacutil.exe로 어셈블리 등록 시 시스템 변경 사항파일 다운로드1
1284정성태5/15/201225699오류 유형: 155. Windows Phone 연결 상태에서 DRIVER POWER STATE FAILURE 블루 스크린 뜨는 현상
1283정성태5/12/201233311.NET Framework: 317. C# 관점에서의 Observer 패턴 구현 [1]파일 다운로드1
1282정성태5/12/201226107Phone: 6. Windows Phone 7 Silverlight에서 Google Map 사용하는 방법 [3]파일 다운로드1
1281정성태5/9/201233194.NET Framework: 316. WPF/Silverlight의 그래픽 단위와 Anti-aliasing 처리를 이해하자 [1]파일 다운로드1
1280정성태5/9/201226158오류 유형: 154. Could not load type 'System.ServiceModel.Activation.HttpModule' from assembly 'System.ServiceModel, ...'.
1279정성태5/9/201224918.NET Framework: 315. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 [1]파일 다운로드1
1278정성태5/8/201226149오류 유형: 153. Visual Studio 디버깅 - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...