里氏替换原则

在说原则之前先回忆几个定义:
一、重写与重载的区别:
1)重写(Override):
1.1)发生在父子类中,方法名相同,参数列表相同,方法体不同
1.2)重写遵循"运行期"绑定,看对象的类型来调用方法
2**)重载(Overload)?*
2.1)发生在一个类中,方法名相同,参数列表不同,方法体不同
2.2)重载遵循"编译期"绑定,看引用的类型来绑定方法

二、继承具有传递性
子类可以重写(覆盖)继承自父类的方法,即方法名和参数列表与父类的方法相同;
子类利用重写修改父类的方法。当子类对象的重写方法被调用时(无论是通过子类
的引用调用还是通过父类的引用调用),运行的是子类的重写修改后的版本

三、重写两同两小一大
两同:方法名和参数列表(方法签名)必须相同
两小:返回值类型、声明异常 比(被重写的)父类的小(或一样)
一大:访问修饰符比(被重写的)父类的大(或一样)

下面开始正题:
里氏替换原则定义
第一种定义:如果对每一个类型为S的对象o1,都有类型为T的对象o2,使得以T定义的所有程序P在所有的对象o1都代换成o2时,程序P的行为没有发生变化,那么类型S是类型T的子类型。
第二种定义:所有引用基类的地方必须能透明地使用其子类的对象。

第二种定义比较通俗,容易理解:只要有父类出现的地方,都可以用子类来替代,而且不会出现任何错误和异常。但是反过来则不行,有子类出现的地方,不能用其父类替代。

四层含义
里氏替换原则对继承进行了规则上的约束,这种约束主要体现在四个方面:

1. 子类必须实现父类的抽象方法,但不得重写(覆盖)父类的非抽象(已实现)方法。

public class Father {

    public void demo(int a){
        System.out.println("Father的demo方法"+a);
    }
}

public class Son extends Father{
        public void demo(int a){
            System.out.println("Son的demo方法"+a);
        }
    }
    
public class TestDemo{
        public static void main(String[] args) {
            Father father = new Father();
            father.demo(1);
            Son son = new Son();
            son.demo(2);
        }
    }

运行结果:
在这里插入图片描述
这里子类把父类方法重写(覆盖)了,这就没有遵循里氏替换原则,遵循的话应该是输出两个“Father的demo方法1”才对

2. 子类中可以增加自己特有的方法。


public class Father {

    public void demo(int a){
        System.out.println("Father的demo方法"+a);
    }
}

public class Son extends Father{
        public void newdemo(int a){
            System.out.println("Son的demo方法"+a);
        }
    }
    
 public class TestDemo{
        public static void main(String[] args) {
            Father father = new Father();
            father.demo(1);
            Son son = new Son();
            son.demo(2);
            son.newdemo(3);
        }
    }

运行结果:
在这里插入图片描述
子类没有覆盖父类,添加了自己的新类,这里属于里氏替换原则。

3. 当子类覆盖或实现父类的方法时,方法的前置条件(即方法的形参)要比父类方法的输入参数更宽松。

public class Father {

    public void demo(HashMap hashMap){
        System.out.println("Father的demo方法");
    }
}

   public class Son extends Father{
        public void demo(Map map){
            System.out.println("Son的demo方法");
        }
    }
    
   public class TestDemo{
        public static void main(String[] args) {
            HashMap hashMap = new HashMap();
            Father father = new Father();
            father.demo(hashMap);
            Son son = new Son();
            son.demo(hashMap);
        }
    }

运行结果:
在这里插入图片描述
注:此处子类并非重写了父类的方法,而是重载了父类的方法。因为子类和父类的方法的输入参数是不同的。子类方法的参数Map比父类方法的参数HashMap的范围要大,所以当参数输入为HashMap类型时,只会执行父类的方法,不会执行子类的重载方法。这符合里氏替换原则(不明白的看上面定义)

4. 当子类的方法实现父类的抽象方法时,方法的后置条件(即方法的返回值)要比父类更严格。

public abstract class Father {
    public abstract Map fun();
}
  public class Son extends Father{
        @Override
        public HashMap fun(){
           HashMap hashMap = new HashMap();
           hashMap.put("子类","子类被执行");
           return hashMap;
        }
    }
  public class TestDemo{
        public static void main(String[] args) {
            Father father = new Son();
            System.out.println(father);
        }
    }

运行结果:
在这里插入图片描述
若在继承时,子类的方法返回值类型范围比父类的方法返回值类型范围大,在子类重写该方法时编译器会报错。

参考:https://www.jianshu.com/p/cf9f3c7c0df5

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值