Lovingly Plagiarized Prepared
by Ryan Haugh
A class should only do one thing
class Customer
{
public void Add()
{
try
{
// Logic goes here
}
catch (Exception ex)
{
System.IO.File.WriteAllText(@"C:\Error.txt", ex.ToString());
}
}
}
What if we want to change our logging implementation?
class Customer
{
private readonly ILogger _logger;
public Customer(ILogger logger)
{
_logger = logger;
}
public void Add()
{
try
{
// Logic goes here
}
catch (Exception ex)
{
_logger.error(ex.ToString());
}
}
}
Software entities (classes, methods, etc.) should be open for extension, but closed for modification
class Customer
{
public int CustType { get; set; }
public decimal CalculateSalePrice(decimal subtotal)
{
if (CustType == 1)
{
return subtotal * 0.90;
}
else
{
return subtotal * 0.95;
}
}
}
What if we want to add a new customer type?
class Customer
{
public virtual decimal CalculateSalePrice(decimal subtotal)
{
return subtotal;
}
}
class SilverCustomer : Customer
{
public override decimal CalculateSalePrice(decimal subtotal)
{
return subtotal * 0.95;
}
}
class GoldCustomer : Customer
{
public override decimal CalculateSalePrice(decimal subtotal)
{
return subtotal * 0.90;
}
}
Subtypes must be substitutable for their base types
class Inquiry : Customer
{
public override decimal CalculateSalePrice(decimal subtotal)
{
return subtotal * 0.95;
}
public override void Add()
{
throw new NotImplementedException();
}
}
What if we want to iterate over a list of customers and add them to our system?
List<Customer> customers = new List<Customer>();
customers.Add(new SilverCustomer());
customers.Add(new GoldCustomer());
customers.Add(new Inquiry());
foreach (Customer customer in customers)
{
customer.Add();
}
interface IDiscount {
decimal CalculateSalePrice(decimal subtotal);
}
interface IPersistence {
void Add();
}
class Inquiry : IDiscount {
public decimal CalculateSalePrice(decimal subtotal) {
return subtotal * 0.95;
}
}
class Customer : IDiscount, IPersistence {
public virtual void Add() {
// Logic goes here
}
public virtual decimal CalculateSalePrice(decimal subtotal) {
return subtotal;
}
}
Clients should not be forced to depend upon interfaces that they do not use
interface IBird
{
void Eat();
void Walk();
void Chirp();
void Fly();
}
What about an ostrich?
interface IBird
{
void Eat();
void Walk();
void Chirp();
}
interface IFlyingBird : IBird
{
void Fly();
}
High level modules should not depend upon low level modules.
Rather, both should depend upon abstractions.
class Car
{
private BasicEngine _engine;
private BasicSteeringWheel _steeringWheel;
public Car ()
{
_engine = new BasicEngine();
_steeringWheel= new BasicSteeringWheel();
}
}
What if we want to swap out the engine and steering wheel?
class Car
{
private IEngine _engine;
private ISteeringWheel _steeringWheel;
public Car (IEngine engine, ISteeringWheel steeringWheel)
{
_engine = engine;
_steeringWheel = steeringWheel;
}
}
class BasicEngine : IEngine {}
class SuperFastEngine : IEngine {}
class BasicSteeringWheel: ISteeringWheel {}
class SuperDeluxeSteeringWheel: ISteeringWheel {}
Car basicRide = new Car(new BasicEngine(), new BasicSteeringWheel());
Car fancyRide = new Car(new SuperFastEngine(), new SuperDeluxeSteeringWheel());