C# 9.0 - (4) 원시 크기 정수(Native ints)
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
32비트 환경에서는 4바이트, 64비트 환경에서는 8바이트로 정의되는 nint, nuint 정수 타입이 새롭게 추가되었습니다.
using System;
class Program
{
static void Main(string[] args)
{
nint x1 = 3;
nuint x2 = 4;
unsafe
{
Console.WriteLine(sizeof(nint));
Console.WriteLine(sizeof(nuint));
}
}
}
/* 출력 결과
- 32비트의 경우
4
4
- 64비트의 경우
8
8
*/
기존에는 위와 같이 플랫폼 의존적인 타입을 갖는 네이티브 모듈들과 연동하기 위해서는 IntPtr/UIntPtr을 사용해야 했지만, 엄밀히 해당 자료형이 포인터가 아닌 경우도 있으므로 이를 위해 새로운 타입을 추가해 다양한 산술 연산자의 활용이 가능하도록 했습니다.
참고로, 비주얼 스튜디오에서 F12(Go To Definition)로 이동하면 각각 IntPtr, UIntPtr 타입으로 이동하는 것을 볼 수 있는데 실제로 내부적인 구현은 그것들에 대응합니다. 그렇긴 하지만 명시적으로 다음의 메서드들은 사용하지 못하도록 막았습니다.
// https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-9.0/native-integers
public static IntPtr MaxValue { get; }
public static IntPtr MinValue { get;}
public static readonly IntPtr Zero; // use 0 instead
public static int Size { get; } // use sizeof() instead
public static IntPtr Add(IntPtr pointer, int offset);
public static IntPtr Subtract(IntPtr pointer, int offset);
public int ToInt32();
public long ToInt64();
public void* ToPointer();
따라서 nint.Zero와 같은 식의 코드는 사용할 수 없습니다. 반면, IntPtr/UIntPtr의 경우에는 명시적으로 포인터 연산이라는 특징으로 포인터끼리 더하는 연산은 제공하지 않지만,
{
IntPtr value1 = new (5); // Target-typed new
IntPtr value2 = new (6);
IntPtr value3 = value1 + value2; // Error CS0019 Operator '+' cannot be applied to operands of type 'IntPtr' and 'IntPtr'
}
nint, nuint는 이러한 연산이 가능합니다.
{
nint value1 = 5; // IntPtr처럼 new 할 필요가 없음.
nint value2 = 6;
nint value3 = value1 + value2;
nint value4 = value1 - value2;
nint value5 = value1 / value2;
nint value6 = value1 * value2;
nint value7 = value1 % value2;
}
(
첨부 파일은 이 글의 예제 코드를 포함합니다.)
시간 되시면 다음의 문서도! ^^
Native-Sized Number Types
; https://github.com/dotnet/corefxlab/blob/master/docs/specs/nativesized.md
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]