<Java设计模式>——里斯替换原则(LSP)小米插排更好用

什么里斯替换原则


定义1:如果对每一个类型为S的对象o1,都有类型T的对象o2,在程序P中,使得T定义的o1都能替代成o2,而程序P的行为没有发生变化,那么类型S是类型T的子类型。
定义2:所有应用基类的地方必须能够透明地使用其子类的对象。

2个定义都是可以的,只是第2种读起来比较绕口,第2个比较简洁,个人建议多品一品定义1。

里斯替换规则


  1. 其他类应该依赖父类或接口
//学生
class Student{
    public String name;
}

//老师
class Teacher{
    //报名
    public static void signUp(Student student){
         System.out.println("当前报名学生姓名:"+student.name);
    }
}

//学生张三
class ZhangSan extends Student{
}

class TeacherWang{
  //报名
    public static void signUp(ZhangSan zhangSan){
         System.out.println("当前报名学生姓名:"+zhangSan.name);
    }
}

看看上面的区别,Teacher依赖的是Student,而TeacherWang依赖的是的ZhangSan,可以说TeacherWang是不符合LSP原则,当然我们在设计程序的时候,并不是全部都要使用这个原则,有的时候也是直接依赖子类的,这个要看具体的需求。

  1. 子类必须完全是父类的方法
Class Student{
    //获取学生证
    public abstract String getStudentID();
}

Class ZhangSan extends Student{
     public String getStudentID(){
        return null;
    }
}

我们看ZhangSan是Student的子类,但是没有实现getStudentID()方法,假如学校门卫是根据学生证,让你走进校园,如果没有学生证,那就没有资格进入校园,那就相当于不是学生。

  1. 子类可以有不同于父类的方法和属性
    ZhangSan、LiSi都是Student,那么他们就有所有Student的属性和方法,但是ZhangSan喜欢下象棋,LiSi喜欢打篮球,那么这样也是可以的,那么反过来讲就不成立了,难道只要是Student都喜欢打篮球和下象棋吗?

  2. 重载父类的方法,参数类型>=重载参数 或 参数类型 != 重载参数

//父类
class Parent {
    public void test(HashMap hashMap){}
}

//子类
class Child extends Parent{
    @Override
    public void test(HashMap hashMap){}

    public void test(Map map){}

    public void test(String text){}
}

//程序P
class Test{
    public static void main(String[] args){
        //使用父类
        Parent parent = new Parent();
        HashMap hashMap = new HashMap();
        parent.test(hashMap);

     //用子类进行替换
     //Child child = new Child ();
     //HashMap hashMap = new HashMap();
     //child.test(hashMap);
   }
}

//根据定义1,咱们把Parent换成Child,最终调用的还是Parent里面的方法。

反之看看成立吗?

//父类
class Parent {
    
    public void test(Map map){}
}

//子类
class Child extends Parent{
   
    public void test(HashMap hashMap){}
    @Override
    public void test(Map map){}

    public void test(String text){}
}

//程序P
class Test{
    public static void main(String[] args){
        //使用父类
        Parent parent = new Parent();
        HashMap hashMap = new HashMap();
        parent.test(hashMap);

     //用子类进行替换
     //Child child = new Child ();
     //HashMap hashMap = new HashMap();
     //child.test(hashMap);
   }
}
替换之后发现child调用的不是父类的test(Map map)方法,而是自己的test(HashMap hashMap)方法,不符合定义1,替换之后不印象程序P的功能。

优缺点


优点
  • 共享代码,减少工作量,子类共享父类的属性和方法。
  • 提高代码重用性
  • 提高代码的扩展性
缺点
  • 集成是侵入性的,只要继承了父类,那么就必须拥有父类的所有的属性和方法。
  • 降低代码的灵活性,由于继承了父类,那么父类就对子类进行了约束。
  • 增强了耦合性,继承本来就是强耦合性的,父类修改属性和方法的时候,必须需要考虑子类的修改。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,778评论 18 399
  • 1.import static是Java 5增加的功能,就是将Import类中的静态方法,可以作为本类的静态方法来...
    XLsn0w阅读 1,267评论 0 2
  • 面向对象主要针对面向过程。 面向过程的基本单元是函数。 什么是对象:EVERYTHING IS OBJECT(万物...
    sinpi阅读 1,096评论 0 4
  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 3,896评论 1 10
  • 裸辞这三个月的时间,看起来像是一片宁静的海。但其实,海底有暗流涌动,夜深人静时尤为汹涌。不过搞笑的是并没有失眠,却...
    段段不知味阅读 700评论 4 3