Ändern von Regeln für die Kompatibilität
Seit der erstmaligen Veröffentlichung wurde immer versucht, in .NET ein hohes Maß an versions- und implementierungsübergreifender Kompatibilität zu gewährleisten. Obwohl .NET 5 (und .NET Core) sowie höhere Versionen im Vergleich zum .NET Framework als neue Technologie angesehen werden können, gibt es zwei Hauptfaktoren, die eine Umstellung vom .NET Framework auf diese .NET-Variante einschränken:
Eine große Anzahl von Entwicklern hat entweder ursprünglich .NET Framework-Anwendungen entwickelt oder entwickelt diese weiterhin. Sie erwarten ein konsistentes Verhalten in .NET-Implementierungen.
.NET Standard-Bibliotheksprojekte ermöglichen es Entwicklern, Bibliotheken zu erstellen, die auf gemeinsame APIs ausgerichtet sind, die vom .NET Framework sowie von .NET 5 und höheren Versionen (sowie von .NET Core) gemeinsam genutzt werden. Entwickler erwarten, dass sich eine Bibliothek in einer .NET 5-Anwendung genauso verhält wie in einer .NET Framework-Anwendung.
Neben der Kompatibilität zwischen .NET-Implementierungen erwarten Entwickler ein hohes Maß an Kompatibilität zwischen verschiedenen den Versionen einer bestimmten .NET-Implementierung. Insbesondere sollte Code, der für eine frühere Version von .NET Core geschrieben wurde, problemlos in .NET 5 oder höher ausgeführt werden können. Tatsächlich erwarten viele Entwickler, dass die neuen APIs in den neu veröffentlichten Versionen von .NET auch mit den Vorabreleases kompatibel sein sollten, in denen diese APIs eingeführt wurden.
In diesem Artikel werden die Änderungen beschrieben, die die Kompatibilität sowie die Art und Weise beeinflussen, wie das .NET-Team die einzelnen Änderungstypen auswertet. Zu verstehen, wie das .NET-Team mit möglichen Breaking Changes umgeht, ist besonders hilfreich für Entwickler, die Pull Requests öffnen, die das Verhalten bestehender .NET-APIs ändern.
In den folgenden Abschnitten werden die Kategorien der Änderungen an den .NET-APIs und deren Auswirkungen auf die Anwendungskompatibilität beschrieben. Änderungen sind zulässig (✔️), nicht zulässig (❌), oder es muss beurteilt und bewertet werden, wie vorhersehbar, offensichtlich und konsistent das bisherige Verhalten war (❓).
Hinweis
- Bibliotheksentwickler können diese Kriterien nicht nur als Leitfaden für die Bewertung von Änderungen an .NET-Bibliotheken verwenden, sondern auch für die Bewertung von Änderungen an ihren Bibliotheken, die auf mehrere .NET-Implementierungen und -Versionen ausgerichtet sind.
- Weitere Informationen zu Kompatibilitätskategorien, zum Beispiel zur Aufwärts- und Abwärtskompatibilität, finden Sie unter Auswirkungen von Codeänderungen auf die Kompatibilität.
Änderungen am öffentlichen Vertrag
Änderungen in dieser Kategorie modifizieren die öffentliche Oberfläche eines Typs. Die meisten Änderungen in dieser Kategorie sind unzulässig, da sie die Abwärtskompatibilität verletzen (die Fähigkeit einer Anwendung, die mit einer früheren Version einer API entwickelt wurde, ohne Neukompilierung auf einer späteren Version ausgeführt zu werden).
Typen
✔️ ZULÄSSIG: Entfernen einer Schnittstellenimplementierung aus einem Typ, wenn die Schnittstelle bereits durch einen Basistyp implementiert ist
❓ ERFORDERT BEURTEILUNG: Hinzufügen einer neuen Schnittstellenimplementierung zu einem Typ
Dies ist eine zulässige Änderung, da sie bestehende Clients nicht beeinträchtigt. Jede Änderung des Typs muss innerhalb der hier definierten Grenzen für zulässige Änderungen erfolgen, damit die neue Implementierung weiterhin zulässig ist. Extreme Vorsicht ist geboten, wenn Schnittstellen hinzugefügt werden, die die Fähigkeit eines Designers oder Serialisierungsmoduls direkt beeinflussen, Code oder Daten zu generieren, die nicht auf unteren Ebenen verwendet werden können. Ein Beispiel ist die ISerializable-Schnittstelle.
❓ ERFORDERT BEURTEILUNG: Einführen einer neuen Basisklasse
Ein Typ kann in einer Hierarchie zwischen zwei bestehenden Typen eingefügt werden, wenn er keinen neuen abstract-Member einführt oder die Semantik oder das Verhalten bestehender Typen ändert. So wurde beispielsweise in .NET Framework 2.0 die Klasse DbConnection zu einer neuen Basisklasse für SqlConnection, die zuvor direkt von Component abgeleitet wurde.
✔️ ZULÄSSIG: Verschieben eines Typs von einer Assembly zu einer anderen
Die alte Assembly muss mit dem TypeForwardedToAttribute markiert werden muss, das auf die neue Assembly verweist.
✔️ ZULÄSSIG: Ändern des Typs struct in einen Typ
readonly struct
Ein
readonly struct
-Typ darf nicht in einenstruct
-Typ geändert werden.✔️ ZULÄSSIG: Hinzufügen des Schlüsselworts sealed oder abstract zu einem Typ, wenn keine zugänglichen (öffentlichen oder geschützten) Konstruktoren vorhanden sind
✔️ ZULÄSSIG: Erweitern der Sichtbarkeit eines Typs
❌NICHT ZULÄSSIG: Ändern des Namespace oder des Namens eines Typs
❌NICHT ZULÄSSIG: Umbenennen oder Entfernen eines öffentlichen Typs
Dies unterbricht sämtlichen Code, der den umbenannten oder entfernten Typen verwendet.
Hinweis
In seltenen Fällen kann .NET eine öffentliche API entfernen. Weitere Informationen finden Sie unter Entfernen von APIs in .NET. Weitere Informationen zur .NET-Supportrichtlinie finden Sie unter .NET-Supportrichtlinie.
❌NICHT ZULÄSSIG: Ändern des einer Enumeration zugrunde liegenden Typs
Dies ist ein Breaking Change der Kompilierzeit und des Verhaltens und darüber hinaus ein binärer Breaking Change, durch die die Analyse der Attributargumente unterbunden werden könnte.
❌NICHT ZULÄSSIG: Versiegeln eines bisher nicht versiegelten Typs
❌NICHT ZULÄSSIG: Hinzufügen einer Schnittstelle zur Menge der Basistypen einer Schnittstelle
Wenn ein Schnittstelle eine Schnittstelle implementiert, die sie zuvor nicht implementiert hat, werden alle Typen, die die ursprüngliche Version der Schnittstelle implementiert haben, beschädigt.
❓ ERFORDERT BEURTEILUNG: Entfernen eine Klasse aus der Menge der Basisklassen oder einer Schnittstelle aus der Menge der implementierten Schnittstellen
Es gibt eine Ausnahme von der Regel zur Entfernung von Schnittstellen: Sie können die Implementierung einer Schnittstelle hinzufügen, die aus der entfernten Schnittstelle abgeleitet ist. Sie können beispielsweise IDisposable entfernen, wenn der Typ oder die Schnittstelle nun IComponent implementiert, wodurch IDisposable implementiert wird.
❌NICHT ZULÄSSIG: Ändern eines
readonly struct
-Typs in einen struct-TypEin
struct
-Typ darf nicht in einenreadonly struct
-Typ geändert werden.❌NICHT ZULÄSSIG: Ändern eines struct-Typs in einen
ref struct
-Typ (und umgekehrt)❌NICHT ZULÄSSIG: Verringern der Sichtbarkeit eines Typs
Die Sichtbarkeit eines Typs darf jedoch erhöht werden.
Member
✔️ ZULÄSSIG: Erweitern der Sichtbarkeit eines nicht virtuellen Members
✔️ ZULÄSSIG: Hinzufügen eines abstrakten Members zu einem öffentlichen Typ, der keine zugänglichen (öffentlichen oder geschützten) Konstruktoren hat, oder wenn der Typ versiegelt ist
Das Hinzufügen eines abstrakten-Members zu einem Typ mit zugänglichen (öffentlichen oder geschützten) Konstruktoren, der nicht
sealed
ist, ist zulässig.✔️ ZULÄSSIG: Einschränken der Sichtbarkeit eines geschützten Members, wenn der Typ keine zugänglichen (öffentlichen oder geschützten) Konstruktoren hat, oder wenn der Typ versiegelt ist
✔️ ZULÄSSIG: Verschieben eines Members in eine Klasse, die in der Hierarchie höher liegt als der Typ, aus dem er entfernt wurde
✔️ ZULÄSSIG: Hinzufügen oder Entfernen einer Außerkraftsetzung
Die Einführung einer Außerkraftsetzung kann dazu führen, dass vorherige Consumer beim Aufrufen von base die Außerkraftsetzung überspringen.
✔️ ZULÄSSIG: Hinzufügen eines Konstruktors zu einer Klasse, gemeinsam mit einem Konstruktor (ohne Parameter), wenn die Klasse zuvor keine Konstruktoren hatte
Das Hinzufügen eines Konstruktors zu einer Klasse, die bisher keine Konstruktoren hatte, ohne den parameterlosen Konstruktor hinzuzufügen, ist jedoch nicht zulässig.
✔️ ZULÄSSIG: Wechseln von einem
ref readonly
- zu einemref
-Rückgabewert (mit Ausnahme von virtuellen Methoden oder Schnittstellen)✔️ ZULÄSSIG: Entfernen von readonly aus einem Feld, es sei denn, der statische Typ des Felds ist ein veränderlicher Werttyp
✔️ ZULÄSSIG: Aufrufen eines bisher noch nicht definierten neuen Ereignisses
❓ ERFORDERT BEURTEILUNG: Hinzufügen eines neuen Instanzenfelds zu einem Typ
Diese Änderung wirkt sich auf die Serialisierung aus.
❌NICHT ZULÄSSIG: Umbenennen oder Entfernen eines öffentlichen Members oder Parameters
Dies unterbricht sämtlichen Code, der den umbenannten oder entfernten Member oder Parameter verwendet.
Dazu gehört auch das Entfernen oder Umbenennen eines Getters oder Setters aus einer Eigenschaft sowie das Umbenennen oder Entfernen von Enumerationsmembern.
❌NICHT ZULÄSSIG: Hinzufügen eines Members zu einer Schnittstelle
Wenn Sie eine Implementierung bereitstellen, führt das Hinzufügen eines neuen Members zu einer vorhandenen Schnittstelle nicht unbedingt zu Kompilierungsfehlern in Downstream-Assemblys. Allerdings unterstützen nicht alle Sprachen Standard-Schnittstellenmember (DIMs). Außerdem kann die Runtime in einigen Szenarien nicht entscheiden, welcher Standard-Schnittstellenmember aufgerufen werden soll. Aus diesen Gründen wird das Hinzufügen eines Members zu einer vorhandenen Schnittstelle als Breaking Change betrachtet.
❌NICHT ZULÄSSIG: Ändern des Werts einer öffentlichen Konstante oder des Enumerationsmembers
❌NICHT ZULÄSSIG: Ändern des Typs einer Eigenschaft, eines Felds, Parameters oder Rückgabewerts
❌NICHT ZULÄSSIG: Hinzufügen, Entfernen oder Ändern der Reihenfolge von Parametern
❌NICHT ZULÄSSIG: Hinzufügen oder Entfernen der Schlüsselwörter in, out oder ref zu oder aus einem Parameter
❌NICHT ZULÄSSIG: Umbenennen eines Parameters (einschließlich der Änderung seiner Groß-/Kleinschreibung)
Dies gilt aus zwei Gründen als Unterbrechung:
Damit werden spät gebundene Szenarien wie die späte Bindungsfunktion in Visual Basic und dynamic in C# unterbrochen.
Es unterbricht die Quellenkompatibilität, wenn Entwickler benannte Argumente verwenden.
❌NICHT ZULÄSSIG: Wechseln von einem
ref
-Rückgabewert zu einemref readonly
-Rückgabewert❌NICHT ZULÄSSIG: Wechseln von einem
ref readonly
- zu einemref
-Rückgabewert in einer virtuellen Methode oder Schnittstelle❌NICHT ZULÄSSIG: Hinzufügen oder Entfernen von abstract zu oder aus einem Member
❌NICHT ZULÄSSIG: Entfernen des Schlüsselworts virtual aus einem Member
❌NICHT ZULÄSSIG: Hinzufügen des Schlüsselworts virtual zu einem Member
Obwohl dies oft keinen Breaking Change darstellt, da der C#-Compiler dazu tendiert, zum Aufrufen nicht virtueller Methoden Intermediate Language (IL)-Anweisungen callvirt auszugeben (
callvirt
führt eine Prüfung auf NULL durch, was bei einem normalen Aufruf nicht der Fall ist), ist dieses Verhalten aus verschiedenen Gründe nicht unveränderlich:C# ist nicht die einzige Sprache, auf die .NET ausgerichtet ist.
Der C#-Compiler versucht zunehmend,
callvirt
für einen normalen Aufruf zu optimieren, wenn die Zielmethode nicht virtuell und wahrscheinlich nicht Null ist (z.B. eine Methode, auf die über den NULL-bedingten Operator „?“ zugegriffen wird).
Aus einer Methode eine virtuelle Methode zu machen bedeutet, dass der Consumercode sie oft nicht virtuell aufruft.
❌NICHT ZULÄSSIG: Umwandeln eines virtuellen Members in einen abstrakten Member
Ein virtueller Member stellt eine Methodenimplementierung bereit, die von einer abgeleiteten Klasse außer Kraft gesetzt werden kann. Ein abstrakter Member stellt keine Implementierung bereit und muss außer Kraft gesetzt werden.
❌NICHT ZULÄSSIG: Hinzufügen des Schlüsselworts sealed zu einem Schnittstellenmember
Durch Hinzufügen von
sealed
zum Standardschnittstellenmember wird dieser nicht virtuell, sodass verhindert wird, dass die Implementierung eines abgeleiteten Typs dieses Members aufgerufen wird.❌NICHT ZULÄSSIG: Hinzufügen eines abstrakten Members zu einem öffentlichen Typ mit zugänglichen (öffentlichen oder geschützten) Konstruktoren, der außerdem nicht versiegelt ist
❌NICHT ZULÄSSIG: Hinzufügen oder Entfernen des Schlüsselworts static zu oder aus einem Member
❌NICHT ZULÄSSIG: Hinzufügen einer Überladung, die eine vorhandene Überladung ausschließt und ein anderes Verhalten definiert
Dies unterbricht vorhandene Clients, die an die vorherige Überladung gebunden waren. Wenn eine Klasse beispielsweise eine einzelne Version einer Methode hat, die UInt32 akzeptiert, wird ein vorhandener Consumer erfolgreich an diese Überladung gebunden, wenn ein Int32-Wert übergeben wird. Wenn Sie jedoch eine Überladung hinzufügen, die Int32 akzeptiert, bindet sich der Compiler beim Neukompilieren oder beim Verwenden der späten Bindung an die neue Überladung. Wenn dies ein anderes Verhalten zur Folge hat, ist dies ein Breaking Change.
❌NICHT ZULÄSSIG: Hinzufügen eines Konstruktors zu einer Klasse, die bisher keinen Konstruktor hatte, ohne den parameterlosen Konstruktor hinzuzufügen
❌NICHT ZULÄSSIG: Hinzufügen von readonly (schreibgeschützt) zu einem Feld
❌NICHT ZULÄSSIG: Verringern der Sichtbarkeit eines Members
Dazu gehört das Reduzieren der Sichtbarkeit eines geschützten Members, wenn zugängliche (
public
oderprotected
) Konstruktoren vorhanden sind und der Typ nicht versiegelt ist. Wenn dies nicht der Fall ist, ist das Reduzieren der Sichtbarkeit eines geschützten Members zulässig.Die Sichtbarkeit eines Members darf erhöht werden.
❌NICHT ZULÄSSIG: Ändern des Typs eines Members
Der Rückgabewert einer Methode oder der Typ einer Eigenschaft oder eines Feldes kann nicht geändert werden. So kann beispielsweise die Signatur einer Methode, die ein Object zurückgibt, nicht geändert werden, um ein String zurückzugeben, oder umgekehrt.
❌NICHT ZULÄSSIG: Hinzufügen eines Instanzfelds zu einer Struktur ohne öffentliche Felder
Wenn eine Struktur nur über öffentliche Felder verfügt oder keine Felder aufweist, können Aufrufer lokale Variablen dieses Strukturtyps deklarieren, ohne den Konstruktor der Struktur aufzurufen oder zuerst die lokale Variable in
default(T)
zu initialisieren, solange alle öffentlichen Felder vor der ersten Verwendung auf die Struktur festgelegt werden. Das Hinzufügen neuer Felder – ob öffentlich oder nicht öffentlich – zu einer solchen Struktur ist ein Breaking Change für diese Aufrufer, da nun für den Compiler die zusätzlichen Felder initialisiert werden müssen.Darüber hinaus ist das Hinzufügen neuer Felder – ob öffentlich oder nicht öffentlich – zu einer Struktur ohne Felder oder nur mit öffentlichen Feldern ein binärer Breaking Change für Aufrufer, die
[SkipLocalsInit]
auf ihren Code angewendet haben. Da der Compiler diese Felder zur Kompilierungszeit nicht berücksichtigt hat, konnte er IL ausgeben, mit der die Struktur nicht vollständig initialisiert wird, was dazu führt, dass die Struktur aus nicht initialisierten Stapeldaten erstellt wird.Wenn eine Struktur keine nicht öffentlichen Felder enthält, erzwingt der Compiler die Initialisierung bereits über den Konstruktor oder mittels
default(T)
, und das Hinzufügen neuer Instanzfelder stellt keinen Breaking Change dar.❌NICHT ZULÄSSIG: Auslösen eines vorhandenen Ereignisses, das bisher noch nicht ausgelöst wurde
Verhaltensänderung
Assemblys
✔️ ZULÄSSIG: Übertragbarmachen einer Assembly, wenn dieselben Plattformen weiterhin unterstützt werden
❌NICHT ZULÄSSIG: Ändern des Namens einer Assembly
❌NICHT ZULÄSSIG: Ändern des öffentlichen Schlüssels einer Assembly
Eigenschaften, Felder, Parameter und Rückgabewerte
✔️ ZULÄSSIG: Ändern des Werts einer Eigenschaft, eines Felds, eines Rückgabewerts oder eines out-Parameters in einen stärker abgeleiteten Typ
Beispielsweise kann eine Methode, die einen Typ von Object zurückgibt, eine String-Instanz zurückgeben. (Allerdings kann nicht die Signatur der Methode nicht geändert werden.)
✔️ ZULÄSSIG: Erhöhen des Bereichs der zulässigen Werte für eine Eigenschaft oder einen Parameter, wenn der Member nicht virtuell ist
Der Bereich der Werte, die an die Methode übergeben werden können oder vom Member zurückgegeben werden, kann erweitert werden, der Parameter- oder der Membertyp jedoch nicht. Beispielsweise können die an eine Methode übergebenen Werte von 0-124 auf 0-255 erweitert werden, der Parametertyp kann jedoch nicht von Byte in Int32 geändert werden.
❌NICHT ZULÄSSIG: Erweitern des Bereichs akzeptierter Werte für eine Eigenschaft oder einen Parameter, wenn der Member virtuell ist
Diese Änderung unterbricht bestehende, außer Kraft gesetzte Member, die für den erweiterten Wertebereich dann nicht korrekt funktionieren.
❌NICHT ZULÄSSIG: Verkleinern des Bereichs akzeptierter Werte für eine Eigenschaft oder einen Parameter
❌NICHT ZULÄSSIG: Erweitern des Bereichs zurückgegebener Werte für eine Eigenschaft, ein Feld, einen Rückgabewert oder den out-Parameter
❌NICHT ZULÄSSIG: Ändern der zurückgegebenen Werte für eine Eigenschaft, ein Feld, einen Rückgabewert einer Methode oder den out-Parameter
❌NICHT ZULÄSSIG: Ändern des Standardwerts einer Eigenschaft, eines Felds oder eines Parameters
Das Ändern oder Entfernen des Standardwerts eines Parameters ist keine binäre Unterbrechung. Das Entfernen des Standardwerts eines Parameters ist eine Quellenunterbrechung, und das Ändern des Standardwerts eines Parameters kann nach der Neukompilierung zu einer Verhaltensunterbrechung führen.
Aus diesem Grund ist das Entfernen von Standardwerten eines Parameters im speziellen Fall des „Verschiebens“ dieser Standardwerte in eine neue Methodenüberladung akzeptabel, um Mehrdeutigkeiten zu beseitigen. Betrachten Sie beispielsweise eine vorhandene Methode
MyMethod(int a = 1)
. Wenn Sie eine Überladung vonMyMethod
mit zwei optionalen Parameterna
undb
einführen, können Sie die Kompatibilität erhalten, indem Sie den Standardwert vona
in die neue Überladung verschieben. Nun sind die beiden ÜberladungenMyMethod(int a)
undMyMethod(int a = 1, int b = 2)
. Dieses Muster ermöglicht die Kompilierung vonMyMethod()
.❌NICHT ZULÄSSIG: Ändern der Genauigkeit eines numerischen Rückgabewerts
❓ ERFORDERT BEURTEILUNG: Eine Änderung im Analysieren von Eingaben und das Auslösen neuer Ausnahmen (auch wenn das Analyseverhalten nicht in der Dokumentation angegeben ist)
Ausnahmen
✔️ ZULÄSSIG: Auslösen einer Ausnahme, die stärker abgeleitet ist als eine vorhandene
Da die neue Ausnahme eine Unterklasse einer vorhandenen Ausnahme ist, wird der bisherige Code zur Ausnahmebehandlung die Ausnahme weiterhin behandeln. In .NET Framework 4 wurden bei der Kulturerstellung und bei Abrufmethoden eine CultureNotFoundException anstelle einer ArgumentException ausgelöst, wenn die Kultur nicht gefunden werden konnte. Da sich CultureNotFoundException aus ArgumentException ableitet, ist dies eine zulässige Änderung.
✔️ ZULÄSSIG: Auslösen einer spezifischeren Ausnahme als NotSupportedException, NotImplementedException, NullReferenceException
✔️ ZULÄSSIG: Auslösen einer Ausnahme, die als nicht behebbar gilt
Nicht behebbare Ausnahmen sollten nicht abgefangen werden, sondern von einem übergeordneten Catch-All-Handler behandelt werden. Daher wird nicht erwartet, dass Benutzer über Code verfügen, der diese expliziten Ausnahmen abfängt. Nicht behebbare Ausnahmen sind:
✔️ ZULÄSSIG: Auslösen einer neuen Ausnahme in einem neuen Codepfad
Die Ausnahme darf nur für einen neuen Codepfad gelten, der mit neuen Parameterwerten oder einen neuen Zustand ausgeführt wird, und der nicht mit vorhandenem Code ausgeführt werden kann, der auf die vorherige Version abzielt.
✔️ ZULÄSSIG: Entfernen eine Ausnahme, um ein stabileres Verhalten oder neue Szenarios zu ermöglichen
So kann beispielsweise eine
Divide
-Methode, die bisher nur positive Werte behandelt und andernfalls ein ArgumentOutOfRangeException ausgelöst hat, so geändert werden, dass sowohl negative als auch positive Werte unterstützt werden, ohne eine Ausnahme auszulösen.✔️ ZULÄSSIG: Ändern des Texts einer Fehlermeldung
Entwickler sollten sich nicht auf den Text von Fehlermeldungen verlassen, die sich auch aufgrund der Kultur des Benutzers ändern.
❌NICHT ZULÄSSIG: Auslösen einer Ausnahme in allen Fällen, die oben nicht aufgeführt sind
❌NICHT ZULÄSSIG: Entfernen einer Ausnahme in allen Fällen, die oben nicht aufgeführt sind
Attribute
✔️ ZULÄSSIG: Ändern des Werts eines Attributs, das nicht überwachbar ist
❌NICHT ZULÄSSIG: Ändern des Werts eines Attributs, das überwachbar ist
❓ ERFORDERT BEURTEILUNG: Entfernen eines Attributs
In den meisten Fällen ist das Entfernen eines Attributs (wie NonSerializedAttribute) ein Breaking Change.
Plattformunterstützung
✔️ ZULÄSSIG: Unterstützen eines Vorgangs auf einer Plattform, der bisher nicht unterstützt wurde
❌NICHT ZULÄSSIG: Keine Unterstützung von oder das jetzige Anfordern eines bestimmten Service Packs für einen Vorgang, der zuvor auf einer Plattform unterstützt wurde
Änderungen an der internen Implementierung
❓ ERFORDERT BEURTEILUNG: Ändern der Oberfläche eines internen Typs
Derartige Änderungen sind im Allgemeinen zulässig, auch wenn sie die private Reflektion unterbrechen. In einigen Fällen, in denen beliebte Bibliotheken von Drittanbietern oder eine große Anzahl von Entwicklern von den internen APIs abhängen, sind solche Änderungen möglicherweise nicht zulässig.
❓ ERFORDERT BEURTEILUNG: Ändern der internen Implementierung eines Members
Diese Änderungen sind im Allgemeinen zulässig, auch wenn sie die private Reflektion unterbrechen. In einigen Fällen, in denen der Kundencode häufig von privater Reflexion abhängt, oder in denen die Änderung unbeabsichtigte Nebenwirkungen mit sich bringt, sind diese Änderungen möglicherweise nicht zulässig.
✔️ ZULÄSSIG: Verbessern der Leistung eines Vorgangs
Die Möglichkeit, die Leistung eines Vorgangs zu ändern, ist unerlässlich, aber solche Änderungen können Code unterbrechen, der von der aktuellen Geschwindigkeit eines Vorgangs abhängt. Dies gilt insbesondere für Code, der vom Zeitpunkt des asynchronen Vorgangs abhängt. Die Leistungsänderung sollte keinen Einfluss auf das andere Verhalten der betreffenden API haben; andernfalls ist diese Änderung ein Breaking Change.
✔️ ZULÄSSIG: Indirektes (und häufig negatives) Ändern der Leistung eines Vorgangs
Wenn die betreffende Änderung aus einem sonstigen Grund nicht als Breaking Change eingestuft wird, ist dies zulässig. Häufig müssen Maßnahmen ergriffen werden, die zusätzliche Vorgänge beinhalten können oder neue Funktionen hinzufügen. Dies hat fast immer Auswirkungen auf die Leistung, kann aber unerlässlich sein, damit die betreffende API wie erwartet funktioniert.
❌NICHT ZULÄSSIG: Ändern einer synchronen API in eine asynchrone API (und umgekehrt)
Codeänderungen
✔️ ZULÄSSIG: Hinzufügen von params zu einem Parameter
❌NICHT ZULÄSSIG: Ändern einer Struktur (struct) in eine Klasse (class) und umgekehrt
❌NICHT ZULÄSSIG: Hinzufügen des Schlüsselworts checked zu einem Codeblock
Diese Änderung kann dazu führen, dass Code, der zuvor so ausgeführt wurde, eine OverflowException ausgelöst. Dies ist nicht zulässig.
❌NICHT ZULÄSSIG: Entfernen von params aus einem Parameter
❌NICHT ZULÄSSIG: Ändern der Reihenfolge, in der Ereignisse ausgelöst werden
Entwickler können vernünftigerweise erwarten, dass Ereignisse in der gleichen Reihenfolge ausgelöst werden, und Entwicklercode hängt häufig von der Reihenfolge ab, in der Ereignisse ausgelöst werden.
❌NICHT ZULÄSSIG: Entfernen des Auslösens eines Ereignisses bei einer bestimmten Aktion
❌NICHT ZULÄSSIG: Ändern, wie oft bestimmte Ereignisse aufgerufen werden
❌NICHT ZULÄSSIG: Hinzufügen des FlagsAttribute zu einem Enumerationstyp