![设计模式就该这样学:基于经典框架源码和真实业务场景](https://wfqqreader-1252317822.image.myqcloud.com/cover/758/33114758/b_33114758.jpg)
3.6 里氏替换原则
3.6.1 里氏替换原则的定义
里氏替换原则(Liskov Substitution Principle,LSP)指如果对每一个类型为T1的对象O1,都有类型为T2的对象O2,使得以T1定义的所有程序P在所有对象O1都替换成O2时,程序P的行为没有发生变化,那么类型T2是类型T1的子类型。
定义看上去比较抽象,我们重新解释一下,可以理解为一个软件实体如果适用于一个父类,则一定适用于其子类,所有引用父类的地方必须能透明地使用其子类的对象,子类对象能够替换父类对象,而程序逻辑不变。也可以理解为,子类可以扩展父类的功能,但不能改变父类原有的功能。根据这个理解,我们对里氏替换原则的定义总结如下。
(1)子类可以实现父类的抽象方法,但不能覆盖父类的非抽象方法。
(2)子类中可以增加自己特有的方法。
(3)当子类的方法重载父类的方法时,方法的前置条件(即方法的输入参数)要比父类的方法更宽松。
(4)当子类的方法实现父类的方法时(重写/重载或实现抽象方法),方法的后置条件(即方法的输出/返回值)要比父类的方法更严格或相等。
3.6.2 使用里氏替换原则解决实际问题
在讲开闭原则的时候,我们埋下了一个伏笔。我们在获取折扣价格后重写覆盖了父类的getPrice()方法,增加了一个获取源码的方法getOriginPrice(),这显然违背了里氏替换原则。我们修改一下代码,不应该覆盖getPrice()方法,增加getDiscountPrice()方法。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_52.jpg?sign=1738983075-sKbnkjpuPS7W3xLGi09nlQIb8My2b9Ah-0-7efbca392f5bb35227e5def491668bdd)
使用里氏替换原则有以下优点。
(1)约束继承泛滥,是开闭原则的一种体现。
(2)加强程序的健壮性,同时变更时可以做到非常好的兼容性,提高程序的维护性、可扩展性,降低需求变更时引入的风险。
现在来描述一个经典的业务场景,用正方形、矩形和四边形的关系说明里氏替换原则,我们都知道正方形是一个特殊的长方形,那么可以创建一个长方形的父类Rectangle类,代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_53.jpg?sign=1738983075-ko7oNdEO5z2UQ545ylzRoypm2FPcR9aM-0-5378ea31bb05e2b087e5905b5af04929)
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_54.jpg?sign=1738983075-uheuJxcfpjIM8Z4FVCg6HjzlVhhJ7mFs-0-8613559d3b9443b7fe8b6501c0d14a2c)
创建正方形Square类继承长方形,代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_55.jpg?sign=1738983075-W7v1G7Q7rtX8H2jf0olvCGcj1UO33pPH-0-533127a160f5711722238dbb0f9d8b8a)
在测试类中,创建resize()方法。根据逻辑,长方形的宽应该大于等于高,我们让高一直自增,直到高等于宽变成正方形,代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_56.jpg?sign=1738983075-K6worJJCi8gFpBnVxfrjLgniQEFSQA6m-0-e9cf81bca28f57d6c2bf27ada9c0a4f3)
客户端测试代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_57.jpg?sign=1738983075-u2ztmv99rmBTvy7Mr1wwO5EKk1mRau9V-0-3793758f035473514c337cce0c345d10)
运行结果如下所示。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_58.jpg?sign=1738983075-yd1Az3BUGu2UcsyMVY9fAak4usEn1w83-0-cf74d08eaed19898bf8850692bbfc644)
由运行结果可知,高比宽还大,这在长方形中是一种非常正常的情况。再来看下面的代码,把长方形替换成它的子类正方形,修改客户端测试代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_59.jpg?sign=1738983075-qYvGQ6OyLFpLdaioLcdyiRqXeFlSCmOO-0-56b3d938685704c22baea5688ccf0c9f)
此时,运行出现了死循环,违背了里氏替换原则,在将父类替换为子类后,程序运行结果没有达到预期。因此,代码设计是存在一定风险的。里氏替换原则只存在于父类与子类之间,约束继承泛滥。再来创建一个基于长方形与正方形共同的抽象——四边形QuardRangle接口,代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_60.jpg?sign=1738983075-sS2JU5Oh3YN7Fdf49a21YGdCnkAfWYdz-0-aae06b9ec96d2a0670537c52a8fe7468)
修改长方形Rectangle类的代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_61.jpg?sign=1738983075-CqUSl2P58rFOwEG006szBOOnxVwTqJMD-0-8a876780d3f940aec5f818acf1c842ae)
修改正方形Square类的代码如下。
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_62.jpg?sign=1738983075-pabwZdXiAqGGmYMZaxwkQb0mez29DtzF-0-f12db6ffce22693e3d651a27e0b21d66)
![img](https://epubservercos.yuewen.com/A1F36C/17725769807799506/epubprivate/OEBPS/Images/txt004_63.jpg?sign=1738983075-tXbmqeOYii2osbj4ktrUjZFyNQUbS99K-0-8eacad9b70f5a1dbc47c1c6a0fdede84)
此时,如果把resize()方法的参数换成四边形QuardRangle类,方法内部就会报错。因为正方形已经没有了setWidth()和setHeight()方法,所以,为了约束继承泛滥,resize()方法的参数只能用长方形Rectangle类。