【愚公系列】2021年12月 面向对象设计原则(二)-开放闭合原则(Open-Closed Principle or OCP)

2021-12-03 16:06:55 浏览数 (1)

文章目录

  • 前言
  • 一、开放闭合原则(Open-Closed Principle or OCP)
  • 二、使用步骤
    • 示例
  • 总结

前言

常用的面向对象设计原则有七个,这七大设计原则都是以可维护性和可复用性为基础的,这些原则并不是孤立存在的,它们相互依赖相互补充,遵循这些设计原则可以有效地提高系统的复用性,同时提高系统的可维护性。

提示:以下是本篇文章正文内容,下面案例可供参考

一、开放闭合原则(Open-Closed Principle or OCP)

开放闭合原则又叫开闭原则,即软件实体应当对扩展开放,对修改封闭。 开闭原则就是指软件实体应当尽量保证在不修改原有代码的情况下,对软件进行扩展。开闭原则是面向对象设计的基石。

二、使用步骤

示例

代码语言:javascript复制
public interface IMobilePhone {

    decimal Price { get; set; }
    string Model { get; set; }
    Color Color { get; set; }

}
代码语言:javascript复制
public enum Color {
    Black,
    White
}

首先用IMobilePhone接口建立手机契约,并向外暴露3个属性,Price属性为手机价格,Model属性为手机型号,Color属性为手机外观颜色,接下来我们用此接口实现一个ApplePhoneX的类。

注:手机接口IMobilePhone最好不要命名为IPhone,ApplePhoneX类不要命名为IPhoneX,因为这容易引起误解。

代码语言:javascript复制
public class ApplePhoneX : IMobilePhone {

    public virtual double Price {
        get => 8799;
        set => Price = value;
    }

    public virtual string Model {
        get => "IPhone X";
        set => Model = value;
    }

    public virtual Color Color {
        get => Color.Black;
        set => Color = value;
    }

}

以下是一个调用方可能的代码:

代码语言:javascript复制
IMobilePhone mobilePhone = new ApplePhoneX();
var price = mobilePhone.Price;

现在需求发生了变化,因为IPhone9上市在即,库克决定为IPhoneX打折促销,黑色的IPhoneX降价为6500.00元,白色的IPhoneX降价为6450.00元, 容易想到的一个做法是,修改IMobilePhone接口,增加DiscountPrice属性,可能如下所示:

代码语言:javascript复制
public interface IMobilePhone {

    double Price { get; set; }
    string Model { get; set; }
    Color Color { get; set; }
    double DiscountPrice { get; set; }//增加

}
代码语言:javascript复制
public class ApplePhoneX : IMobilePhone {

    public virtual double Price {
        get => 8799;
        set => Price = value;
    }

    public virtual string Model {
        get => "IPhone X";
        set => Model = value;
    }

    public virtual Color Color {
        get => Color.Black;
        set => Color = value;
    }

    public virtual double DiscountPrice {//增加
        get => Color == Color.Black ? 6500.00 : 6450.00;
        set => DiscountPrice = value;
    }

}
代码语言:javascript复制
public class HuaweiPhone : IMobilePhone {
    //需要修改
}
代码语言:javascript复制
public class SmartisanPhone : IMobilePhone {
    //需要修改
}

但是这次修改将会影响到所有实现IMobilePhone接口的类,比如HuaweiPhone类和SmartisanPhone类。接口作为一种契约,应当是一种稳定的存在,不允许轻易修改,否则将明显违反开闭原则。以下给出一个解决方案以供参考:

代码语言:javascript复制
public class DiscountApplePhoneX : ApplePhoneX {

    public override double Price {
        get => Color == Color.Black ? 6500.00 : 6450.00;
        set => Price = value;
    }

}
代码语言:javascript复制
IMobilePhone mobilePhone = new DiscountApplePhoneX();
var price = mobilePhone.Price;

总结

通过增加一个继承自ApplePhoneX的DiscountApplePhoneX类并重写Price方法来解决这个新需求,原来的所有代码均不需要更改,只要在使用打折手机的地方修改其使用即可,符合开闭原则。

0 人点赞