Former-commit-id: fd56571888259555122d8a0f58c68838229cea2b
11 KiB
Breaking Change Rules
Behavioral Changes
Property, Field, Parameter and Return Values
✓ Allowed
- Increasing the range of accepted values for a property or parameter if the member is not
virtual
Note that the range can only increase to the extent that it does not impact the static type. e.g. it is OK to remove
if (x > 10) throw new ArgumentOutOfRangeException("x")
, but it is not OK to change the type ofx
fromint
tolong
.
- Returning a value of a more derived type for a property, field, return or
out
value
Note, again, that the static type cannot change. e.g. it is OK to return a
string
instance where anobject
was returned previously, but it is not OK to change the return type fromobject
tostring
.
✗ Disallowed
- Increasing the range of accepted values for a property or parameter if the member is
virtual
This is breaking because any existing overridden members will now not function correctly for the extended range of values.
-
Decreasing the range of accepted values for a property or parameter, such as a change in parsing of input and throwing new errors (even if parsing behavior is not specified in the docs)
-
Increasing the range of returned values for a property, field, return or
out
value -
Changing the returned values for a property, field, return or 'out' value, such as the value returned from
ToString
If you had an API which returned a value from 0-10, but actually intended to divide the value by two and forgot (return only 0-5) then changing the return to now give the correct value is a breaking.
-
Changing the default value for a property, field or parameter (either via an overload or default value)
-
Changing the value of an enum member
-
Changing the precision of a numerical return value
Exceptions
✓ Allowed
- Throwing a more derived exception than an existing exception
For example,
CultureInfo.GetCultureInfo(String)
used to throwArgumentException
in .NET Framework 3.5. In .NET Framework 4.0, this was changed to throwCultureNotFoundException
which derives fromArgumentException
, and therefore is an acceptable change.
- Throwing a more specific exception than
NotSupportedException
,NotImplementedException
,NullReferenceException
or an exception that is considered unrecoverable
Unrecoverable exceptions should not be getting caught and will be dealt with on a broad level by a high-level catch-all handler. Therefore, users are not expected to have code that catches these explicit exceptions. The list of unrecoverable exceptions are: *
StackOverflowException
*SEHException
*ExecutionEngineException
*AccessViolationException
-
Throwing a new exception that only applies to a code-path which can only be observed with new parameter values, or state (that couldn't hit by existing code targeting the previous version)
-
Removing an exception that was being thrown when the API allows more robust behavior or enables new scenarios
For example, a Divide method which only worked on positive values, but threw an exception otherwise, can be changed to support all values and the exception is no longer thrown.
✗ Disallowed
-
Throwing a new exception in any other case not listed above
-
Removing an exception in any other case not listed above
Platform Support
✓ Allowed
- An operation previously not supported on a specific platform, is now supported
✗ Disallowed
- An operation previously supported on a specific platform is no longer supported, or now requires a specific service-pack
Code
✓ Allowed
- A change which is directly intended to increase performance of an operation
The ability to modify the performance of an operation is essential in order to ensure we stay competitive, and we continue to give users operational benefits. This can break anything which relies upon the current speed of an operation, sometimes visible in badly built code relying upon asynchronous operations. Note that the performance change should have no affect on other behavior of the API in question, otherwise the change will be breaking.
- A change which indirectly, and often adversely, affects performance
Assuming the change in question is not categorized as breaking for some other reason, this is acceptable. Often, actions need to be taken which may include extra operation calls, or new functionality. This will almost always affect performance, but may be essential to make the API in question function as expected.
- Changing the text of an error message
Not only should users not rely on these text messages, but they change anyways based on culture
- Calling a brand new event that wasn't previously defined.
✗ Disallowed
- Adding the
checked
keyword to a code-block
This may cause code in a block to to begin to throwing exceptions, an unacceptable change.
- Changing the order in which events are fired
Developers can reasonably expect events to fire in the same order.
-
Removing the raising of an event on a given action
-
Changing a synchronous API to asynchronous (and vice versa)
-
Firing an existing event when it was never fired before
-
Changing the number of times given events are called
Source and Binary Compatibility Changes
Assemblies
✓ Allowed
- Making an assembly portable when the same platforms are still supported
✗ Disallowed
-
Changing the name of an assembly
-
Changing the public key of an assembly
Types
✓ Allowed
-
Adding the
sealed
orabstract
keyword to a type when there are no accessible (public or protected) constructors -
Increasing the visibility of a type
-
Introducing a new base class
So long as it does not introduce any new abstract members or change the semantics or behavior of existing members, a type can be introduced into a hierarchy between two existing types. For example, between .NET Framework 1.1 and .NET Framework 2.0, we introduced
DbConnection
as a new base class forSqlConnection
which previously derived fromComponent
.
- Adding an interface implementation to a type
This is acceptable because it will not adversely affect existing clients. Any changes which could be made to the type being changed in this situation, will have to work within the boundaries of acceptable changes defined here, in order for the new implementation to remain acceptable. Extreme caution is urged when adding interfaces that directly affect the ability of the designer or serializer to generate code or data, that cannot be consumed down-level. An example is the
ISerializable
interface.
-
Removing an interface implementation from a type when the interface is already implemented lower in the hierarchy
-
Moving a type from one assembly into another assembly
The old assembly must be marked with
TypeForwardedToAttribute
pointing to the new location
✗ Disallowed
-
Adding the
sealed
orabstract
keyword to a type when there are accessible (public or protected) constructors -
Decreasing the visibility of a type
-
Removing the implementation of an interface on a type
It is not breaking when you added the implementation of an interface which derives from the removed interface. For example, you removed
IDisposable
, but implementedIComponent
, which derives fromIDisposable
.
-
Removing one or more base classes for a type, including changing
struct
toclass
and vice versa -
Changing the namespace or name of a type
Members
✓ Allowed
-
Adding an abstract member to a public type when there are no accessible (
public
orprotected
) constructors, or the type issealed
-
Moving a member onto a class higher in the hierarchy tree of the type from which it was removed
-
Increasing the visibility of a member that is not
virtual
-
Decreasing the visibility of a
protected
member when there are no accessible (public
orprotected
) constructors or the type issealed
-
Changing a member from
abstract
tovirtual
-
Adding
virtual
to a member
Make note, that marking a member virtual might cause previous consumers to still call the member non-virtually.
- Introducing or removing an override
Make note, that introducing an override might cause previous consumers to skip over the override when calling
base
.
✗ Disallowed
-
Adding an member to an interface
-
Adding an abstract member to a type when there are accessible (
public
orprotected
) constructors and the type is notsealed
-
Adding a constructor to a class which previously had no constructor, without also adding the default constructor
-
Adding an overload that precludes an existing overload, and defines different behavior
This will break existing clients that were bound to the previous overload. For example, if you have a class that has a single version of a method that accepts a
uint
, an existing consumer will successfully bind to that overload, if simply passing anint
value. However, if you add an overload that accepts anint
, recompiling or via late-binding the application will now bind to the new overload. If different behavior results, then this is a breaking change.
-
Removing or renaming a member, including a getter or setter from a property or enum members
-
Decreasing the visibility of a
protected
member when there are accessible (public
orprotected
) constructors and the type is notsealed
-
Adding or removing
abstract
from a member -
Removing the
virtual
keyword from a member -
Adding or removing
static
keyword from a member
Signatures
✓ Allowed
-
Adding
params
to a parameter -
Removing
readonly
from a field, unless the static type of the field is a mutable value type
✗ Disallowed
-
Adding
readonly
to a field -
Adding the
FlagsAttribute
to an enum -
Changing the type of a property, field, parameter or return value
-
Adding, removing or changing the order of parameters
-
Removing
params
from a parameter -
Adding or removing
out
orref
keywords from a parameter -
Renaming a parameter (including case)
This is considered breaking for two reasons:
-
It breaks late-bound scenarios, such as Visual Basic's late-binding feature and C#'s
dynamic
-
It breaks source compatibility when developers use named parameters.
-
Changing a parameter modifier from
ref
toout
, or vice versa
Attributes
✓ Allowed
- Changing the value of an attribute that is not observable
✗ Disallowed
- Removing an attribute
Although this item can be addressed on a case to case basis, removing an attribute will often be breaking. For example,
NonSerializedAttribute
- Changing values of an attribute that is observable