七个原则6-里氏替换原则
-
定义:如果对每一个类型为
T1
的对象o1
,都有类型为T2
的对象o2
,使得以T1
定义的所有程序P
在所有的对象o1
都替换成o2
时,程序P
的行为没有发生变化,那么类型T2
是类型T1
的子类型 - 定义扩展:一个软件实体如果适用一个父类的话,那一定适用于子类,所有引用父类的地方必须能透明地使用其子类的对象,子类对象能够替换父类对象,而程序逻辑不变
-
引申意义:子类可以扩展父类的功能,但不能改变父类原有的功能
- 含义1:子类可以实现父类的抽象方法,但不能覆盖父类的非抽象方法
- 含义2:子类可以增加自己特有的方法
- 含义3:当子类方法重载父类的方法时,方法的前置条件(即方法的输入/入参)要比父类的输入参数跟宽松
- 含义4:当子类的方法实现父类的方法时(重写/重载或实现抽象方法),方法的后置条件(即方法的输出/返回值)要比父类更严格或相等
-
优点:
- 优点1:约束继承泛滥,开闭原则的一种体现
- 优点2:加强程序的健壮性,同时变更时也可以做到非常好的兼容性;提高程序的维护性、扩展性。降低需求变更时引入的风险
示例1:矩形是特殊的正方形
矩形是一种特殊的长方形,当我们设计类时如果使 Square
继承 Rectangle
,这又是不合适的,因为 Square
的 Length 和 Width 一致,它不需要两个 set 方法。所以合理的方式应该是让他们共同实现 四边形 Quadrangle
接口
示例2:子类重载父类方法时,方法入参要比父类宽松
public class Base {
public void method(HashMap map){
System.out.println("父类被执行");
}
}
//------------------------------------------
public class Child extends Base{
@Override
public void method(HashMap map) {
System.out.println("子类HashMap入参方法被执行");
}
public void method(Map map){
System.out.println("子类Map入参方法被执行");
}
}
//--------------------------------------------
public class Test {
public static void main(String[] args) {
Base child = new Child();
HashMap hashMap = new HashMap<>();
child.method(hashMap);
}
}
示例3:子类重载父类方法时,方法返回值要比父类严格
public interface Base {
Map method();
}
//-------------------------------------------------
public class Child implements Base {
@Override
public HashMap method() {
HashMap hashMap = new HashMap();
System.out.println("子类method被执行");
hashMap.put("message","子类method被执行");
return hashMap;
}
}
//---------------------------------------------------
public class Test {
public static void main(String[] args) {
Child child = new Child();
System.out.println(child.method());
}
}
网友评论