호환성 규칙 변경
지금까지 .NET은 버전 간은 물론 .NET 구현 전체에서 높은 수준의 호환성을 유지해 왔습니다. .NET 5(및 .NET Core) 이상 버전은 .NET Framework에 비해 새로운 기술로 간주될 수 있지만 다음 두 가지 주요 요인으로 인해 .NET Framework에서 이러한 .NET 구현을 완전히 분리하기는 어렵습니다.
많은 개발자가 원래 .NET Framework 애플리케이션을 개발했거나 계속 개발하고 있습니다. 이러한 개발자는 .NET 구현 전체에서 일관된 동작을 기대합니다.
.NET Standard 라이브러리 프로젝트를 사용하면 개발자가 .NET Framework와 .NET 5(및 .NET Core) 이상 버전에서 공유하는 공통 API를 대상으로 하는 라이브러리를 만들 수 있습니다. 개발자는 .NET 5 애플리케이션에서 사용되는 라이브러리가 .NET Framework 애플리케이션에서 사용되는 동일한 라이브러리와 똑같이 동작할 것으로 기대합니다.
개발자는 .NET 구현 전체의 호환성과 함께 .NET의 특정 구현 버전 전체에서 높은 수준의 호환성을 기대합니다. 특히 이전 버전의 .NET Core용으로 작성된 코드가 .NET 5 또는 이상 버전에서 원활하게 실행되어야 합니다. 실제로 대부분의 개발자는 새로 릴리스된 .NET 버전의 새 API가 해당 API를 도입한 시험판 버전과도 호환되어야 한다고 기대합니다.
이 문서에서는 호환성에 영향을 주는 변경 내용과 .NET 팀이 각 변경 유형을 평가하는 방식에 대해 간략하게 설명합니다. .NET 팀이 호환성이 손상되는 가능한 변경에 접근하는 방법을 이해하는 것은 기존 .NET API의 동작을 수정하는 끌어오기 요청을 여는 개발자에게 특히 유용합니다.
다음 섹션에서는 .NET API의 변경 범주 및 애플리케이션 호환성에 미치는 영향을 설명합니다. 변경 내용은 허용(✔️)️ 또는 허용되지 않음(❌)으로 적용되고, 이전 동작이 얼마나 예측 가능하고 명백하며 일관성이 있었는지(❓)에 대한 판단과 평가가 필요합니다.
참고
- .NET 라이브러리의 변경 내용을 평가하는 방법에 대한 가이드 역할을 할 뿐 아니라, 라이브러리 개발자는 이러한 기준을 사용하여 여러 .NET 구현과 버전을 대상으로 하는 고유한 라이브러리의 변경 내용을 평가할 수도 있습니다.
- 호환성 범주(예: 이후 버전과 이전 버전의 호환성)에 대한 자세한 내용은 코드 변경 내용이 호환성에 미치는 영향을 참조하세요.
공용 계약 수정
이 범주의 변경 내용은 형식의 공용 노출 영역을 ‘수정’합니다. 이 범주의 변경 내용은 대부분 이전 버전과의 호환성(이전 버전의 API로 개발된 애플리케이션이 최신 버전에서 다시 컴파일하지 않고도 실행되는 기능)을 위반하기 때문에 허용되지 않습니다.
유형
✔️ 허용: 인터페이스가 기본 형식을 통해 이미 구현된 경우 형식에서 인터페이스 구현 제거
❓ 판단 필요: 형식에 새 인터페이스 구현 추가
이 변경 내용은 기존 클라이언트에 부정적인 영향을 주지 않으므로 허용됩니다. 새 구현이 허용되려면 형식의 모든 변경 내용이 여기서 정의된 허용되는 변경 내용의 경계를 넘지 않아야 합니다. 하위 수준에서 사용할 수 없는 코드 또는 데이터를 생성하는 디자이너 또는 직렬 변환기의 기능에 직접 영향을 주는 인터페이스를 추가할 때는 각별한 주의가 필요합니다. 예를 들어 ISerializable 인터페이스가 있습니다.
❓ 판단 필요: 새 기본 클래스 도입
새 추상 구성원을 도입하거나 기존 형식의 의미 체계 또는 동작을 변경하지 않는 경우, 계층 구조에서 두 기존 형식 사이에 형식을 도입할 수 있습니다. 예를 들어 .NET Framework 2.0에서는 DbConnection 클래스가 이전에 Component에서 직접 파생된 SqlConnection의 새 기본 클래스가 되었습니다.
✔️ 허용: 어셈블리 간에 형식 이동
이전 어셈블리는 새 어셈블리를 가리키는 TypeForwardedToAttribute로 표시해야 합니다.
✔️ 허용: 구조체 형식을
readonly struct
형식으로 변경readonly struct
형식을struct
형식으로 변경할 수는 없습니다.✔️ 허용: 형식의 표시 유형 확장
❌허용 안 함: 형식의 네임스페이스 또는 이름 변경
❌허용 안 함: public 형식의 이름 바꾸기 또는 제거
이 경우 이름이 바뀌었거나 제거된 형식을 사용하는 모든 코드가 중단됩니다.
참고 항목
드문 경우로 .NET이 공용 API를 제거할 수 있습니다. 자세한 내용은 .NET에서 API 제거를 참조하세요. .NET 지원 정책에 관한 자세한 내용은 .NET 지원 정책을 참조하세요.
❌허용 안 함: 열거형의 기본 형식 변경
특성 인수를 구문 분석할 수 없게 만드는 호환성이 손상되는 이진 변경일 뿐만 아니라 호환성이 손상되는 컴파일 시간 및 동작 변경입니다.
❌허용 안 함: 이전에 봉인되지 않은 형식 봉인
❌허용 안 함: 인터페이스의 기본 형식 집합에 인터페이스 추가
인터페이스가 이전에 구현하지 않은 인터페이스를 구현하는 경우, 원래 버전의 인터페이스를 구현한 모든 형식이 중단됩니다.
❓ 판단 필요: 기본 클래스 세트에서 클래스 제거 또는 구현된 인터페이스 세트에서 인터페이스 제거
인터페이스 제거 규칙의 한 가지 예외로, 제거한 인터페이스에서 파생되는 인터페이스 구현을 추가할 수 있습니다. 예를 들어 이제 IDisposable을 구현하는 IComponent를 형식 또는 인터페이스가 구현하는 경우 IDisposable을 제거할 수 있습니다.
❌허용 안 함:
readonly struct
형식을 구조체 형식으로 변경struct
형식은readonly struct
형식으로 변경할 수 있습니다.❌허용 안 함: 구조체 형식을
ref struct
형식으로 변경하거나 그 반대로 변경❌허용 안 함: 형식의 표시 유형 축소
단, 형식의 표시 유형을 늘릴 수는 있습니다.
멤버
✔️ 허용: 가상이 아닌 구성원의 표시 유형 확장
✔️ 허용: 액세스할 수 있는(공용 또는 보호된) 생성자가 없거나 봉인된 형식인 경우 공용 형식에 추상 구성원 추가
단, 액세스할 수 있는(public 또는 protected) 생성자가 있고
sealed
가 아닌 형식에는 추상 멤버를 추가할 수 없습니다.✔️ 허용: 형식에 액세스할 수 있는(공용 또는 보호된) 생성자가 없거나 봉인된 형식인 경우 보호된 구성원의 표시 유형 제한
✔️ 허용: 구성원이 제거된 형식보다 계층 구조의 상위 클래스로 해당 구성원 이동
✔️ 허용: 재정의 추가 또는 제거
재정의를 도입하는 경우 base를 호출하면 이전 소비자가 재정의를 건너뛸 수 있습니다.
✔️ 허용: 이전에 클래스에 생성자가 없었던 경우 매개 변수가 없는 생성자와 함께 클래스에 생성자 추가
단, 매개 변수가 없는 생성자를 추가하지 ‘않고’ 이전에 생성자가 없었던 클래스에 생성자를 추가할 수는 없습니다.
✔️ 허용:
ref readonly
에서ref
반환 값으로 변경(가상 메서드 또는 인터페이스 제외)✔️ 허용: 필드의 정적 형식이 변경할 수 있는 값 형식이 아닌 경우 필드에서 readonly 제거
✔️ 허용: 이전에 정의되지 않았던 새 이벤트 호출
❓ 판단 필요: 형식에 새 인스턴스 필드 추가
이 변경 내용은 serialization에 영향을 줍니다.
❌허용 안 함: public 멤버 또는 매개 변수 이름 바꾸기 또는 제거
이 경우 이름이 바뀌었거나 제거된 멤버 또는 매개 변수를 사용하는 모든 코드가 중단됩니다.
여기에는 속성에서 getter 또는 setter 제거 또는 이름 바꾸기, 열거형 구성원 이름 바꾸기 또는 제거가 포함됩니다.
❌허용 안 함: 인터페이스에 멤버 추가
구현을 제공하는 경우 기존 인터페이스에 새 멤버를 추가한다고 해서 반드시 다운스트림 어셈블리에서 컴파일 오류가 발생하는 것은 아닙니다. 그러나 모든 언어가 DIM(기본 인터페이스 멤버)을 지원하는 것은 아닙니다. 또한 일부 시나리오에서는 런타임에서 호출할 기본 인터페이스 멤버를 결정할 수 없습니다. 이러한 이유로 기존 인터페이스에 멤버를 추가하는 것은 호환성이 손상되는 변경으로 간주됩니다.
❌허용 안 함: public 상수 또는 열거형 멤버 값 변경
❌허용 안 함: 속성, 필드, 매개 변수 또는 반환 값의 형식 변경
❌허용 안 함: 매개 변수 추가, 제거 또는 순서 변경
❌허용 안 함: 매개 변수 이름 바꾸기(대/소문자 변경 포함)
다음 두 가지 이유로 호환성이 손상되는 변경으로 간주됩니다.
❌허용 안 함:
ref
반환 값에서ref readonly
반환 값으로 변경❌허용 안 함: 가상 메서드 또는 인터페이스의
ref readonly
에서ref
반환 값으로 변경❌허용 안 함: 멤버에서 abstract 추가 또는 제거
❌허용 안 함: 멤버에서 virtual 키워드 제거
❌허용 안 함: 멤버에 virtual 키워드 추가
C# 컴파일러는 비가상 메서드를 호출하기 위해 callvirt IL(중간 언어) 명령을 내보내는 경향이 있으므로(
callvirt
는 일반 호출과 달리 null 검사를 수행함) 호환성이 손상되는 변경이 아닌 경우가 많지만, 이 동작은 다음과 같은 몇 가지 이유로 가변적입니다..NET의 대상 언어가 C#만은 아닙니다.
대상 메서드가 비가상이고 null이 아닐 수 있는 경우(예: ?. null 전파 연산자를 통해 액세스한 메서드), C# 컴파일러는 항상
callvirt
를 일반 호출에 최적화하려고 합니다.
메서드를 virtual로 설정하면 소비자 코드에서 비가상적으로 메서드를 호출하게 되는 경우가 자주 발생합니다.
❌허용 안 함: 가상 멤버를 추상으로 설정
가상 멤버는 파생 클래스에서 재정의할 수 있는 메서드 구현을 제공합니다. 추상 멤버는 구현을 제공하지 않으므로 재정의해야 합니다.
❌허용 안 함: 인터페이스 멤버에 sealed 키워드 추가
기본 인터페이스 멤버에
sealed
를 추가하면 가상이 아닌 것으로 표시되어 파생 형식의 해당 멤버 구현이 호출되지 않습니다.❌허용 안 함: 액세스할 수 있는(public 또는 protected) 생성자가 있고 sealed가 아닌 public 형식에 추상 멤버 추가
❌허용 안 함: 멤버에서 static 키워드 추가 또는 제거
❌허용 안 함: 기존 오버로드를 차단하고 다른 동작을 정의하는 오버로드 추가
이전 오버로드에 바인딩된 기존 클라이언트의 호환성이 손상됩니다. 예를 들어 클래스에 UInt32를 허용하는 메서드의 단일 버전이 있는 경우, 기존 소비자가 Int32 값을 전달할 때 해당 오버로드에 성공적으로 바인딩됩니다. 그러나 Int32를 허용하는 오버로드를 추가하면, 다시 컴파일하거나 런타임에 바인딩을 사용할 때 이제 컴파일러가 새 오버로드에 바인딩됩니다. 다른 동작이 발생하는 경우 호환성이 손상되는 변경입니다.
❌허용 안 함: 매개 변수가 없는 생성자를 추가하지 않고 이전에 생성자가 없었던 클래스에 생성자 추가
❌허용 안 함: 필드에 readonly 추가
❌허용 안 함: 멤버의 표시 유형 축소
여기에는 형식에 액세스할 수 있는(
public
또는protected
) 생성자가 있고 봉인된 형식이 아닌 경우 보호된 구성원의 표시 유형을 줄이는 경우가 포함됩니다. 이러한 경우가 아니면 보호된 멤버의 표시 유형을 줄일 수 있습니다.구성원의 표시 유형은 늘릴 수 있습니다.
❌허용 안 함: 멤버의 형식 변경
메서드의 반환 값이나 속성 또는 필드의 형식을 수정할 수 없습니다. 예를 들어 Object가 반환되는 메서드의 시그니처를 String이 반환되도록 변경할 수 없으며, 그 반대의 경우도 마찬가지입니다.
❌허용 안 함: nonpublic 필드가 없는 구조체에 인스턴스 필드 추가
구조체에 public 필드만 있거나 필드가 전혀 없는 경우 처음 사용하기 전에 구조체에 대해 모든 public 필드가 설정되기만 하면, 호출자는 구조체의 생성자를 호출하거나 먼저 로컬을
default(T)
로 초기화하지 않고도 해당 구조체 형식의 로컬을 선언할 수 있습니다. 이제 컴파일러에서 추가 필드를 초기화해야 하므로 이러한 구조체에 public 또는 nonpublic인 새 필드를 추가하는 것은 이러한 호출자에 대한 소스가 손상되는 변경입니다.또한 필드가 없거나 public 필드만 있는 구조체에 public 또는 nonpublic인 새 필드를 추가하는 것은 코드에
[SkipLocalsInit]
를 적용한 호출자에 대한 이진이 손상되는 변경입니다. 컴파일러는 컴파일 시간에 이러한 필드를 인식하지 못했기 때문에 구조체를 완전히 초기화하지 않는 IL을 내보내 초기화되지 않은 스택 데이터에서 구조체를 만들 수 있습니다.구조체에 nonpublic 필드가 있는 경우 컴파일러는 이미 생성자 또는
default(T)
를 통해 초기화를 적용하며 새 인스턴스 필드를 추가하는 것은 호환성이 손상되는 변경이 아닙니다.❌허용 안 함: 이전에는 발생하지 않았던 시기에 기존 이벤트 발생
동작 변경
어셈블리
✔️ 허용: 동일한 플랫폼이 여전히 지원될 때 어셈블리를 이식 가능으로 설정
❌허용 안 함: 어셈블리 이름 변경
❌허용 안 함: 어셈블리의 공개 키 변경
속성, 필드, 매개 변수 및 반환 값
✔️ 허용: 속성, 필드, 반환 값 또는 out 매개 변수의 값을 더 파생된 형식으로 변경
예를 들어 Object 형식을 반환하는 메서드는 String 인스턴스를 반환할 수 있습니다. 단, 메서드 시그니처는 변경할 수 없습니다.
✔️ 허용: 구성원이 가상이 아닌 경우 속성 또는 매개 변수에 허용되는 값의 범위 확장
메서드에 전달할 수 있거나 구성원에서 반환되는 값의 범위는 확장할 수 있지만, 매개 변수 또는 구성원 형식은 확장할 수 없습니다. 예를 들어 메서드에 전달되는 값은 0~124에서 0~255로 확장할 수 있지만, 매개 변수 형식은 Byte에서 Int32로 변경할 수 없습니다.
❌허용 안 함: 멤버가 virtual인 경우 속성 또는 매개 변수에 허용되는 값의 범위 확장
이 변경으로 인해 확장된 값 범위에서 제대로 작동하지 않는 재정의된 기존 멤버의 호환성이 손상됩니다.
❌허용 안 함: 속성 또는 매개 변수에 허용되는 값의 범위 축소
❌허용 안 함: 속성, 필드, 반환 값 또는 out 매개 변수의 반환 값 범위 확장
❌허용 안 함: 속성, 필드, 메서드 반환 값 또는 out 매개 변수의 반환 값 변경
❌허용 안 함: 속성, 필드 또는 매개 변수의 기본값 변경
parameter 기본값을 변경하거나 제거하는 것은 이진 중단이 아닙니다. 매개 변수 기본값을 제거하는 것은 소스 중단에 해당하며 매개 변수 기본값을 변경하면 다시 컴파일한 후 동작이 중단될 수 있습니다.
이러한 이유로 매개 변수 기본값을 제거하는 것은 모호성을 제거하기 위해 해당 기본값을 새 메서드 오버로드로 “이동”하는 특정 경우에 허용됩니다. 예를 들어 기존 메서드
MyMethod(int a = 1)
를 고려해 보세요. 두 개의 선택적 매개 변수a
및b
를 사용하여MyMethod
의 오버로드를 도입하면 기본값a
를 새 오버로드로 이동하여 호환성을 유지할 수 있습니다. 이제 두 오버로드는MyMethod(int a)
및MyMethod(int a = 1, int b = 2)
입니다. 이 패턴을 사용하면MyMethod()
를 컴파일할 수 있습니다.❌허용 안 함: 숫자 반환 값의 전체 자릿수 변경
❓ 판단 필요: 입력 구문 분석 및 새 예외 throw 변경(구문 분석 동작이 문서에 지정되지 않은 경우 포함)
예외
✔️ 허용: 기존 예외보다 더 파생된 예외 throw
새 예외가 기존 예외의 서브클래스이기 때문에, 이전 예외 처리 코드에서 계속 예외를 처리합니다. 예를 들어 .NET Framework 4에서는 문화권을 찾을 수 없는 경우 문화권 만들기 및 검색 메서드가 ArgumentException 대신 CultureNotFoundException을 throw하기 시작했습니다. CultureNotFoundException은 ArgumentException에서 파생되기 때문에 허용되는 변경입니다.
✔️ 허용: NotSupportedException, NotImplementedException, NullReferenceException보다 더 구체적인 예외 throw
✔️ 허용: 복구할 수 없는 것으로 간주되는 예외 throw
복구할 수 없는 예외는 catch하지 않고, 높은 수준의 범용 처리기에서 처리해야 합니다. 따라서 사용자는 이러한 명시적 예외를 catch하는 코드를 사용할 수 없습니다. 복구할 수 없는 예외는 다음과 같습니다.
✔️ 허용: 새 코드 경로에서 새 예외 throw
새 매개 변수 값 또는 상태로 실행되었으며, 이전 버전을 대상으로 하는 기존 코드에서 실행할 수 없는 새 코드 경로에만 예외를 적용해야 합니다.
✔️ 허용: 더욱 강력한 동작이나 새로운 시나리오를 사용하기 위해 예외 제거
예를 들어 이전에는 양수 값만 처리하고, 양수 값이 아닐 경우 ArgumentOutOfRangeException을 throw한
Divide
메서드에서 예외를 throw하지 않고 음수 값과 양수 값을 모두 지원하도록 변경할 수 있습니다.✔️ 허용: 오류 메시지의 텍스트 변경
개발자는 사용자의 문화권에 따라 변경되는 오류 메시지의 텍스트에 의존해서는 안 됩니다.
❌허용 안 함: 위에 나열되지 않은 다른 모든 경우의 예외 throw
❌허용 안 함: 위에 나열되지 않은 다른 모든 경우의 예외 제거
특성
✔️ 허용: 식별할 수 없는 특성 값 변경
❌허용 안 함: 관찰할 수 ‘있는’ 특성 값 변경
❓ 판단 필요: 특성 제거
대부분의 경우, NonSerializedAttribute와 같은 특성 제거는 호환성이 손상되는 변경입니다.
플랫폼 지원
✔️ 허용: 플랫폼에서 이전에 지원되지 않았던 작업 지원
❌허용 안 함: 이전에 플랫폼에서 지원된 작업에 대해 특정 서비스 팩 요구 또는 지원 안 함
내부 구현 변경
❓ 판단 필요: 내부 형식의 노출 영역 변경
해당 변경은 프라이빗 리플렉션의 호환성이 손상되는 변경이지만 일반적으로 허용됩니다. 인기 있는 타사 라이브러리 또는 많은 개발자가 내부 API를 사용하는 경우에는 이러한 변경이 허용되지 않을 수 있습니다.
❓ 판단 필요: 구성원의 내부 구현 변경
이러한 변경은 프라이빗 리플렉션의 호환성이 손상되는 변경이지만 일반적으로 허용됩니다. 고객 코드가 프라이빗 리플렉션에 자주 종속되거나, 이 변경으로 인해 의도하지 않은 부작용이 도입되는 경우에는 이러한 변경이 허용되지 않을 수 있습니다.
✔️ 허용: 작업 성능 향상
작업 성능을 수정하는 기능은 필수지만, 이러한 변경으로 인해 현재 작업 속도에 의존하는 코드의 호환성이 손상될 수 있습니다. 비동기 작업 타이밍에 종속된 코드의 경우 특히 그렇습니다. 성능 변경은 해당 API의 다른 동작에 영향을 주지 않아야 합니다. 영향을 주면 호환성이 손상되는 변경이 됩니다.
✔️ 허용: 간접적으로(때로는 부정적으로) 작업 성능 변경
해당 변경이 다른 이유로 호환성이 손상되는 변경으로 분류되지 않은 경우에는 허용됩니다. 추가 작업을 포함할 수 있거나 새 기능을 추가하는 조치를 수행해야 하는 경우가 많습니다. 이 변경은 거의 항상, 성능에 영향을 주지만 해당 API가 예상대로 작동하는 데 필요할 수 있습니다.
❌허용 안 함: 동기 API를 비동기로 변경하거나 그 반대로 변경
코드 변경 내용
✔️ 허용: 매개 변수에 params 추가
❌허용 안 함: 코드 블록에 checked 키워드 추가
이 변경으로 인해 이전에 실행되었던 코드에서 OverflowException이 throw될 수 있으므로 허용되지 않습니다.
❌허용 안 함: 매개 변수에서 params 제거
❌허용 안 함: 이벤트 발생 순서 변경
개발자는 이벤트가 동일한 순서로 발생할 것으로 기대할 수 있으며, 개발자 코드가 이벤트 발생 순서에 종속된 경우가 많습니다.
❌허용 안 함: 지정된 동작에서 이벤트 발생 제거
❌허용 안 함: 지정된 이벤트 호출 횟수 변경
❌허용 안 함: 열거형 형식에 FlagsAttribute 추가
추가 정보
.NET