Properties should be stateless with respect to other properties, i.e. there should not be a difference between first setting property A
and then B
or vice versa.
Using a method is way better than a property, if:
- The work is more expensive than setting a field value;
- it represents a conversion such as the
Object.ToString()
method; - it returns a different result each time it is called, even if the arguments didn't change. For example, the
NewGuid()
method returns a different value each time it is called; or - the operation causes a side effect such as changing some internal state not directly related the property (which violates the Command Query Separation).
Populating an internal cache or implementing lazy-loading is a good exception.
Having properties that cannot be used at the same time typically signals a type that is representing two conflicting concepts. Even though those concepts may share some of the behavior and state, they obviously have different rules that do not cooperate.
This violation is often seen in domain models and introduces all kinds of conditional logic related to those conflicting rules, causing a ripple effect that significantly worsens the maintenance burden.
Based on SRP, a method have a single responsibility.
A stateful object is an object that contains many properties and lots of behavior behind that. If you expose such an object through a static property or method of some other object, it will be very difficult to refactor or unit test a class that relies on such a stateful object. In general, introducing a construction like that is a great example of violating many of the guidelines of this document.
A classic example of this is the HttpContext.Current
property, part of ASP.NET. Many see the HttpContext
class as a source for a lot of ugly code. In fact, the testing guideline Isolate the Ugly Stuff often refers to this class.
In general, you don't want callers to be able to change an internal collection, so don't return arrays, lists or other collection classes directly. Instead, return an IEnumerable<T>
, or, if the caller must be able to determine the count, an ICollection<T>
.
In .NET 4.5, you can also use IReadOnlyCollection<T>
, IReadOnlyList<T>
or IReadOnlyDictionary<TKey, TValue>
.
Returning null
can be unexpected by the caller. Always return an empty collection or an empty string instead of a null
reference. This also prevents cluttering your code base with additional checks for null
, or even worse, String.IsNotNullOrEmpty()
or String.IsNullOrWhiteSpace()
.
If your member needs a specific piece of data, define parameters as specific as that and don't take a container object instead. For instance, consider a method that needs a connection string that is exposed through some central IConfiguration
interface. Rather than taking a dependency on the entire configuration, just define a parameter for the connection string. This not only prevents unnecessary coupling, it also improved maintainability in a long run.
Instead of using strings, integers and decimals for representing domain specific types such as an ISBN number, an email address or amount of money, consider created dedicated value objects that wrap both the data and the validation rules that apply to it. By doing this, you prevent ending up having multiple implementations of the same business rules, which both improves maintainability and prevents bugs.