C# 형식 시스템

C#은 강력한 형식의 언어입니다. 모든 변수 및 상수에는 값으로 계산되는 모든 식을 실행하는 형식이 있습니다. 모든 메서드 선언은 각 입력 매개 변수와 반환 값의 이름, 형식, 종류(값, 참조 또는 출력)를 지정합니다. .NET 클래스 라이브러리는 기본 제공 숫자 형식과 다양한 구문을 나타내는 복합 형식을 정의합니다. 여기에는 파일 시스템, 네트워크 연결, 개체의 컬렉션과 배열, 날짜가 포함됩니다. 일반 C# 프로그램에서는 클래스 라이브러리의 형식 및 프로그램의 문제 도메인에 관련된 개념을 모델링하는 사용자 정의 형식을 사용합니다.

형식에 저장된 정보에는 다음 항목이 포함될 수 있습니다.

  • 형식 변수에 필요한 스토리지 공간.
  • 형식이 나타낼 수 있는 최대값 및 최소값.
  • 형식에 포함되는 멤버(메서드, 필드, 이벤트 등).
  • 형식이 상속하는 기본 형식.
  • 구현하는 인터페이스.
  • 허용되는 작업 유형.

컴파일러는 형식 정보를 사용하여 코드에서 수행되는 모든 작업의 형식이 안전한지 확인합니다. 예를 들어 int 형식의 변수를 선언하는 경우 컴파일러를 통해 더하기 및 빼기 작업에서 변수를 사용할 수 있습니다. bool 형식의 변수에 대해 같은 작업을 수행하려고 하면 컴파일러는 다음 예제와 같이 오류를 생성합니다.

int a = 5;
int b = a + 2; //OK

bool test = true;

// Error. Operator '+' cannot be applied to operands of type 'int' and 'bool'.
int c = a + test;

참고 항목

C 및 C++ 개발자는 C#에서 bool이(가) int(으)로 변환될 수 없음을 알고 있습니다.

컴파일러는 형식 정보를 실행 파일에 메타데이터로 포함합니다. CLR(공용 언어 런타임)는 런타임에 이 메타데이터를 사용하여 메모리를 할당 및 회수할 때 형식 안정성을 추가로 보장합니다.

변수 선언에서 형식 지정

프로그램에서 변수나 상수를 선언할 때 컴파일러가 형식을 유추하게 하려면 형식을 지정하거나 var 키워드를 사용해야 합니다. 다음 예제에서는 기본 제공 숫자 형식 및 복잡한 사용자 정의 형식을 둘 다 사용하는 일부 변수 선언을 보여 줍니다.

// Declaration only:
float temperature;
string name;
MyClass myClass;

// Declaration with initializers (four examples):
char firstLetter = 'C';
var limit = 3;
int[] source = [0, 1, 2, 3, 4, 5];
var query = from item in source
            where item <= limit
            select item;

메서드 매개 변수 및 반환 값의 형식은 메서드 선언에서 지정됩니다. 다음 시그니처는 입력 인수로 int(이)가 필요하고 문자열을 반환하는 메서드를 보여줍니다.

public string GetName(int ID)
{
    if (ID < names.Length)
        return names[ID];
    else
        return String.Empty;
}
private string[] names = ["Spencer", "Sally", "Doug"];

변수를 선언한 후에는 새 형식으로 다시 선언할 수 없으며 선언된 형식과 호환되지 않는 값을 할당할 수 없습니다. 예를 들어 int을(를) 선언한 다음 여기에 true 부울 값을 할당할 수 없습니다. 그러나 값은 새 변수에 할당되거나 메서드 인수로 전달될 경우 다른 형식으로 변환할 수 있습니다. 데이터 손실을 일으키지 않는 형식 변환은 컴파일러에서 자동으로 수행됩니다. 데이터 손실을 일으킬 수 있는 변환의 경우 소스 코드에 캐스트가 있어야 합니다.

자세한 내용은 캐스팅 및 형식 변환을 참조하세요.

기본 제공 형식

C#은 기본 제공 형식의 표준 집합을 제공합니다. 이러한 표준 집합은 정수, 부동 소수점 값, 부울 식, 텍스트 문자, 10진수 값, 기타 데이터 형식을 나타냅니다. 이 밖에도 기본 제공 stringobject 형식이 있습니다. 이러한 형식을 이러한 형식을 모든 C# 프로그램에서 사용할 수 있습니다. 기본 제공 형식의 전체 목록은 기본 제공 형식을 참조하세요.

사용자 지정 형식

struct, class, interface, enum, record 구문을 사용하여 자체 사용자 지정 형식을 만듭니다. .NET 클래스 라이브러리 자체는 자체 애플리케이션에서 사용할 수 있는 사용자 지정 형식의 컬렉션입니다. 기본적으로 클래스 라이브러리의 가장 자주 사용되는 형식을 모든 C# 프로그램에서 사용할 수 있습니다. 기타 형식은 해당 형식을 정의하는 어셈블리에 프로젝트 참조를 명시적으로 추가하는 경우에만 사용할 수 있습니다. 컴파일러에 어셈블리에 대한 참조가 포함된 후에는 소스 코드에서 해당 어셈블리에 선언된 형식의 변수(및 상수)를 선언할 수 있습니다. 자세한 내용은 .NET 클래스 라이브러리를 참조하세요.

CTS(공용 형식 시스템)

.NET의 형식 시스템에 대한 다음과 같은 두 가지 기초 사항을 이해해야 합니다.

  • 형식 시스템은 상속 원칙을 지원합니다. 형식은 기본 형식이라는 다른 형식에서 파생될 수 있습니다. 파생 형식은 기본 형식의 메서드, 속성 및 기타 멤버를 상속합니다(몇 가지 제한 사항 있음). 기본 형식이 다른 형식에서 파생될 수도 있습니다. 이 경우 파생 형식은 상속 계층 구조에 있는 두 기본 형식의 멤버를 상속합니다. System.Int32(C# 키워드: int)과 같은 기본 제공 숫자 형식을 포함한 모든 형식은 궁극적으로 단일 기본 형식 System.Object(C# 키워드: object)에서 파생됩니다. 이 통합 형식 계층 구조를 CTS(공용 형식 시스템)라고 합니다. C#의 상속에 대한 자세한 내용은 상속을 참조하세요.
  • CTS의 각 형식은 값 형식 또는 참조 형식으로 정의됩니다. 이러한 형식에는 .NET 클래스 라이브러리의 모든 사용자 지정 형식과 자체 사용자 정의 형식도 포함됩니다. struct을(를) 사용하여 정의한 형식은 값 형식이고, 모든 기본 제공 숫자 형식은 structs입니다. class 또는 record 키워드를 사용하여 정의한 형식은 참조 형식입니다. 참조 형식과 값 형식의 컴파일 시간 규칙 및 런타임 동작은 서로 다릅니다.

다음 그림에서는 CTS에서 값 형식과 참조 형식 간의 관계를 보여 줍니다.

Screenshot that shows CTS value types and reference types.

참고 항목

가장 일반적으로 사용되는 형식은 모두 System 네임스페이스에 구성되어 있다는 사실을 알 수 있습니다. 그러나 형식이 포함된 네임스페이스는 형식이 값 형식인지 또는 참조 형식인지와 관련이 없습니다.

클래스 및 구조체는 .NET의 CTS(공용 형식 시스템)의 기본 구문 중 두 가지입니다. 각각은 기본적으로 하나의 논리 단위에 속하는 데이터 및 동작 집합을 캡슐화하는 데이터 구조입니다. 데이터와 동작은 클래스, 구조체 또는 레코드의 ‘멤버’입니다. 멤버는 이 문서의 뒷부분에 나열된 대로 해당 메서드, 속성, 이벤트 등을 포함합니다.

클래스, 구조체 또는 레코드 선언은 런타임에 인스턴스 또는 개체를 만드는 데 사용되는 청사진과도 같습니다. Person이라는 클래스, 구조체 또는 레코드를 정의하는 경우 Person이 형식의 이름입니다. 형식 Person의 변수 p를 선언하고 초기화하면 pPerson의 개체 또는 인스턴스로 지칭됩니다. 같은 Person 형식의 여러 인스턴스를 만들 수 있으며 각 인스턴스는 속성 및 필드에 서로 다른 값을 가질 수 있습니다.

클래스는 참조 형식입니다. 이 형식의 개체가 만들어지면 개체가 할당되는 변수는 해당 메모리에 대한 참조만 보유합니다. 개체 참조가 새 변수에 할당되면 새 변수는 원래 개체를 나타냅니다. 모두 동일한 데이터를 참조하므로 한 변수의 변경 내용이 다른 변수에도 반영됩니다.

구조체는 값 형식입니다. 구조체가 만들어지면 해당 구조체가 할당되는 변수에 구조체의 실제 데이터가 포함됩니다. 구조체를 새 변수에 할당하면 구조체가 복사됩니다. 따라서 새 변수와 원래 변수에 동일한 데이터의 두 가지 별도 복사본이 포함됩니다. 한 복사본의 변경 내용은 다른 복사본에 영향을 주지 않습니다.

레코드 형식은 참조 형식(record class) 또는 값 형식(record struct)일 수 있습니다. 레코드 형식에는 값 같음을 지원하는 메서드가 포함되어 있습니다.

일반적으로 클래스는 더 복잡한 동작을 모델링하는 데 사용됩니다. 클래스는 일반적으로 클래스 개체가 만들어진 후 수정할 데이터를 저장합니다. 구조체는 작은 데이터 구조에 가장 적합합니다. 구조체는 일반적으로 구조체가 만들어진 후 수정하지 않을 데이터를 저장합니다. 레코드 형식은 추가 컴파일러 합성 멤버가 있는 데이터 구조입니다. 레코드는 일반적으로 개체가 만들어진 후 수정하지 않을 데이터를 저장합니다.

값 형식

값 형식은 System.Object에서 파생되는 System.ValueType에서 파생됩니다. System.ValueType에서 파생되는 형식에는 CLR의 특수 동작이 있습니다. 값 형식 변수에는 해당 값이 직접 포함됩니다. 구조체의 메모리는 변수가 선언된 컨텍스트(무엇이든지)에서 인라인으로 할당됩니다. 값 형식 변수에 대한 별도 힙 할당이나 가비지 수집 오버헤드는 없습니다. 값 형식인 record struct 형식을 선언하고 레코드의 합성 멤버를 포함할 수 있습니다.

값 형식에는 structenum의 두 가지 범주가 있습니다.

기본 제공 숫자 형식은 구조체이며, 액세스할 수 있는 필드와 메서드가 있습니다.

// constant field on type byte.
byte b = byte.MaxValue;

하지만 단순 비집계 형식처럼 값을 선언하고 변수에 할당합니다.

byte num = 0xA;
int i = 5;
char c = 'Z';

값 형식은 sealed입니다. 값 형식(예: System.Int32)에서 형식을 파생할 수 없습니다. 구조체는 System.ValueType에서만 상속할 수 있기 때문에 사용자 정의 클래스 또는 구조체에서 상속하는 구조체를 정의할 수 없습니다. 그러나 구조체는 하나 이상의 인터페이스를 구현할 수 있습니다. 구조체 형식을 구현하는 인터페이스 형식으로 캐스팅할 수 있습니다. 이 캐스트로 인해 boxing 작업은 관리되는 힙의 참조 형식 개체 내에 구조체를 래핑합니다. Boxing 작업은 System.Object 또는 인터페이스 형식을 입력 매개 변수로 사용하는 메서드에 값 형식을 전달할 때 발생합니다. 자세한 내용은 boxing 및 unboxing을 참조하세요.

struct 키워드를 사용하여 고유한 사용자 지정 값 형식을 만듭니다. 일반적으로 구조체는 다음 예제와 같이 소규모 관련 변수 집합의 컨테이너로 사용됩니다.

public struct Coords
{
    public int x, y;

    public Coords(int p1, int p2)
    {
        x = p1;
        y = p2;
    }
}

구조체에 대한 자세한 내용은 구조 형식을 참조하세요. 값 형식에 대한 자세한 내용은 값 형식을 참조하세요.

값 형식의 다른 범주는 enum입니다. 열거형은 명명된 정수 상수 집합을 정의합니다. 예를 들어, .NET 클래스 라이브러리의 System.IO.FileMode 열거형에는 파일을 여는 방법을 지정하는 명명된 상수 정수 집합이 포함됩니다. 이 패턴은 다음 예제와 같이 정의됩니다.

public enum FileMode
{
    CreateNew = 1,
    Create = 2,
    Open = 3,
    OpenOrCreate = 4,
    Truncate = 5,
    Append = 6,
}

System.IO.FileMode.Create 상수 값은 2입니다. 그러나 이 이름은 소스 코드를 읽는 사람에게 훨씬 더 의미가 있습니다. 따라서 상수 리터럴 숫자 대신 열거형을 사용하는 것이 더 좋습니다. 자세한 내용은 System.IO.FileMode를 참조하세요.

모든 열거형은 System.ValueType에서 상속받는 System.Enum에서 상속됩니다. 구조체에 적용되는 모든 규칙이 열거형에도 적용됩니다. 열거형에 대한 자세한 내용은 열거형 형식을 참조하세요.

참조 형식

class, record, delegate, 배열 또는 interface로 정의된 형식은 reference type입니다.

reference type의 변수를 선언하는 경우 해당 형식의 인스턴스로 할당하거나 new 연산자를 사용해 생성할 때까지 값 null을 포함합니다. 다음 예제에서는 클래스의 생성 및 할당을 보여줍니다.

MyClass myClass = new MyClass();
MyClass myClass2 = myClass;

interfacenew 연산자를 사용하여 직접 인스턴스화할 수 없습니다. 대신, 인터페이스를 구현하는 클래스의 인스턴스를 만들고 할당합니다. 다음 예제를 참조하세요.

MyClass myClass = new MyClass();

// Declare and assign using an existing value.
IMyInterface myInterface = myClass;

// Or create and assign a value in a single statement.
IMyInterface myInterface2 = new MyClass();

개체가 만들어지면 관리되는 힙에 메모리가 할당됩니다. 변수는 개체의 위치에 대한 참조만 포함합니다. 관리되는 힙의 형식은 할당될 때와 회수될 때 모두 오버헤드가 필요합니다. ‘가비지 수집’은 회수를 수행하는 CLR의 자동 메모리 관리 기능입니다. 그러나 가비지 수집은 고도로 최적화되고 대부분 시나리오에서 성능 문제를 일으키지 않습니다. 가비지 수집에 대한 자세한 내용은 자동 메모리 관리를 참조하세요.

모든 배열은 해당 요소가 값 형식이더라도 참조 형식입니다. 배열은 System.Array 클래스에서 암시적으로 파생됩니다. 다음 예제와 같이 C#에서 제공하는 단순한 구문을 사용하여 배열을 선언하고 사용할 수 있습니다.

// Declare and initialize an array of integers.
int[] nums = [1, 2, 3, 4, 5];

// Access an instance property of System.Array.
int len = nums.Length;

참조 형식은 상속을 완벽하게 지원합니다. 클래스를 만들 때 sealed로 정의되지 않은 기타 인터페이스 또는 클래스에서 상속할 수 있습니다. 기타 클래스는 직접 만든 클래스에서 상속되고 가상 메서드를 재정의할 수 있습니다. 클래스를 직접 만드는 방법에 대한 자세한 내용은 클래스, 구조체, 레코드를 참조하세요. 상속 및 가상 메서드에 대한 자세한 내용은 상속을 참조하세요.

리터럴 값 형식

C#에서는 리터럴 값이 컴파일러에서 형식을 받습니다. 숫자의 끝에 문자를 추가하여 숫자 리터럴의 입력 방법을 지정할 수 있습니다. 예를 들어 값 4.56float로 처리되도록 지정하려면 숫자 뒤에 "f" 또는 "F"를 추가합니다(4.56f). 문자를 추가하지 않으면 컴파일러가 리터럴의 형식을 유추합니다. 문자 접미사를 사용하여 지정할 수 있는 형식에 대한 자세한 내용은 정수 숫자 형식부동 소수점 숫자 형식을 참조하세요.

리터럴은 형식화되고 모든 형식이 궁극적으로 System.Object에서 파생되기 때문에 다음과 같은 코드를 작성하고 컴파일할 수 있습니다.

string s = "The answer is " + 5.ToString();
// Outputs: "The answer is 5"
Console.WriteLine(s);

Type type = 12345.GetType();
// Outputs: "System.Int32"
Console.WriteLine(type);

제네릭 형식

실제 형식(‘구체적 형식’)에 대한 자리 표시자로 사용되는 하나 이상의 ‘형식 매개 변수’를 사용하여 형식을 선언할 수 있습니다. 클라이언트 코드는 형식의 인스턴스를 만들 때 구체적 형식을 제공합니다. 해당 형식을 제네릭 형식이라고 합니다. 예를 들어 .NET 형식 System.Collections.Generic.List<T>에는 변환을 통해 이름 T가 제공되는 하나의 형식 매개 변수가 있습니다. 형식의 인스턴스를 만들 때 목록에 포함될 개체의 형식(예: string)을 지정합니다.

List<string> stringList = new List<string>();
stringList.Add("String example");
// compile time error adding a type other than a string:
stringList.Add(4);

형식 매개 변수를 사용하면 각 요소를 개체로 변환할 필요 없이 같은 클래스를 재사용하여 요소 형식을 포함할 수 있습니다. 컴파일러는 컬렉션 요소의 특정 형식을 인식하며, 예를 들어 이전 예제에서 stringList 개체에 정수를 추가하려는 경우 컴파일 시간에 오류를 발생시킬 수 있기 때문에 제네릭 컬렉션 클래스를 강력한 형식의 컬렉션이라고 합니다. 자세한 내용은 제네릭을 참조하세요.

암시적 형식, 무명 형식 및 nullable 값 형식

var 키워드를 사용하여 클래스 멤버가 아닌 로컬 변수를 암시적으로 형식화할 수 있습니다. 이 변수는 컴파일 타임에 형식을 받지만 형식은 컴파일러에서 제공됩니다. 자세한 내용은 암시적으로 형식화된 지역 변수를 참조하세요.

저장하거나 메서드 경계 외부로 전달할 의도가 없는 관련 값의 단순 집합에 대한 명명된 형식을 만드는 것이 불편할 수 있습니다. 이 목적으로는 무명 형식을 만들 수 있습니다. 자세한 내용은 무명 형식을 참조하세요.

일반적인 값 형식은 null 값을 가질 수 없습니다. 그러나 형식 뒤에 ?를 추가하면 null 허용 값 형식을 만들 수 있습니다. 예를 들어 int?null 값을 가질 수도 있는 int 형식입니다. nullable 값 형식은 제네릭 구조체 형식 System.Nullable<T>의 인스턴스입니다. null 허용 값 형식은 특히 숫자 값이 null일 수 있는 데이터베이스에 데이터를 전달하는 경우에 유용합니다. 자세한 내용은 nullable 값 형식을 참조하세요.

컴파일 시간 형식 및 런타임 형식

변수의 컴파일 시간과 런타임 형식은 서로 다를 수 있습니다. 컴파일 시간 형식은 소스 코드에서 선언되거나 유추되는 변수의 형식입니다. 런타임 형식은 해당 변수에서 참조하는 인스턴스의 형식입니다. 다음 예제에서는 이와 같은 두 가지 유형이 동일한 경우가 많습니다.

string message = "This is a string of characters";

하지만 다음 두 가지 예에서 보듯 컴파일 시간 형식이 다른 경우도 있습니다.

object anotherMessage = "This is another string of characters";
IEnumerable<char> someCharacters = "abcdefghijklmnopqrstuvwxyz";

앞선 두 예제에서 런타임 형식은 string입니다. 컴파일 시간 형식은 첫 번째 줄에서 object, 두 번째 줄에서 IEnumerable<char>입니다.

두 형식이 변수에 대해 다른 경우 컴파일 시간 형식과 런타임 형식이 적용되는 경우를 이해하는 것이 중요합니다. 컴파일 시간 형식에 따라 컴파일러가 수행하는 모든 작업이 결정됩니다. 이러한 컴파일러 동작으로는 메서드 호출 확인, 오버로드 확인 및 사용 가능한 암시적 및 명시적 캐스트가 있습니다. 런타임 형식에 따라 런타임에서 확인되는 모든 작업이 결정됩니다. 이러한 런타임 동작에는 가상 메서드 호출 디스패치, isswitch 식 계산, 기타 형식 테스트 API가 포함됩니다. 코드가 형식과 상호 작용하는 방식을 보다 효과적으로 이해하려면 어떤 작업이 어떤 형식에 적용되는지를 알아야 합니다.

자세한 내용은 다음 문서를 참조하세요.

C# 언어 사양

자세한 내용은 C# 언어 사양을 참조하세요. 언어 사양은 C# 구문 및 사용법에 대 한 신뢰할 수 있는 소스 됩니다.