java的依赖倒置原则你了解多少

网友投稿 236 2022-11-01

java的依赖倒置原则你了解多少

目录依赖倒置原则案例:背景:1.面向实现编程2.面向接口编程(简单版)总结

依赖倒置原则

什么是依赖倒置原则:

高层模块不应该依赖低层模块,二者都应该依赖其抽象

抽象不应该依赖细节,细节应该依赖抽象

针对接口编程,不要针对实现编程

即:

每个类尽量继承自接口或者抽象类

优点:减少类之间的耦合,提高代码的稳定性,代码的可读性维护性。

案例:

背景:

现在有一个用户类叫Ggzx(也就是我),想要学习一些课程,简单的来实现调用学习的方法,然后在一个Test类之中输入学习myAKc的内容。但是我暂时只学java和web,但是可能我后面还要学习Spring,SpringMVC…

1.面向实现编程

public class Ggzx {

public void stduyJava(){

System.out.println("学习了java课程");

}

public void studyWeb(){

System.out.println("学习了Web课程");

}

}

public class Test {

public static void main(String[] args) {

Ggzx ggzx=new Ggzx();

ggzx.studyJava();

ggzx.studypython();

ggzx.studyGo();

}

}

分析:

上面使用的面向实现编程,但是Test作为我们控制的"应用层",也就是高层,而Ggzx作为低层,其实这样在比较简单的例子中,其实是没问题的,因为假如不需要扩展,仅仅是实现两个很简单的功能,并没有必要去面向接口开发,但是一般在开发中通常有很复杂的开发环境和开发需求。

现在如果想添加新的功能,学习其他的课程,怎么办???

继续使用面向实现编程,直接在 Ggzx 类中直接添加新的方法,可以完成这个功能需求。

用上面的方法实现有没有缺点???

1.学习的课程和 Ggzx 类耦合比较严重。是学习的课程只能通过Ggzx 才能得到 。并且是想要学习新的课程也要在 Ggzx 类中不断添加和修改 —>高耦合

2.Ggzx 作为当前 demo 的底层,经常的被改动,高层Test依赖于低层 Ggzx 的实现 ---->对应依赖倒置原myAKc则中的:高层过度依赖低层了

2.面向接口编程(简单版)

为了解决上面出现的问题,我们可以考虑把学习的课程抽出来成为一个类。到现在,类和类之间的耦合其实就已经降低很多了。然后将其当做参数传入Ggzx里面,然后调用课程里面的学习方法

//web课程类

public class WebCourse {

public void studyCourse() {

System.out.println("学习了Web课程");

}

}

//这里是Java课程类

public class JavaCourse {

public void studyCourse() {

System.out.println("学习Java课程");

}

}

当我们写出来这两个类,想要对Ggzx里面的学习方法进行编写的时候,有没有发现其实有一些小问题呢????

Ggzx里面接收这些类的参数是什么??

难道要这样?

//以下是Ggzx类中的内容

public void studyJava(JavaCourse javaCourse){

}

public void studyWeb(WebCourse webCourse){

}

nonono,如果这样做,虽然当前已经把课程类和 Ggzx 用户剥离一点点了,但是是还是形同虚设,课程类虽然分离开了,但是还是像狗皮膏药一样贴在 Ggzx 类中,但是看着还是很难受,高层 Test 调用方法还是得依赖 Ggzx 里面有什么方法

每次加入新课程,都需要修改底层功能

如何修改???

接口是个好东西,课程类之间是不是都包含同样一个方法,被学习的方法( studyCourse ),那么我们可以将所有课程类都实现一个ICourse课程!

对应上面的问题,我们该传入什么参数能解决问题??可以传入一个接口

改编后的 UML 图解展示(Ggzx 被废弃,用新的 NewGgzx 代替):

(如果没了解过UML类图,或者是纯小白,只需要知道一个大框是一个类,虚线表示实现了箭头方向的接口,小m是方法 即可)

观察上面的UML图

WebCourse 和 JavaCourse 实现自同一个接口 ICourse,每个课程都有自己的 studyXxx 方法。

这样好在什么地方?

课程类和Ggzx类是解耦的,无论你增加多少个课程类,只要实现了ICourse接口,都能直接传入Ggzx的studyMyCourse()方法中

public interface ICourse {

void studyCourse();

}

public interface ICourse {

void studyCourse();

}

public class NewGgzx {

public void studyMyCourse(ICourse iCourse){

iCourse.studyCourse();

}

}

上面就是案例的面向接口编程,我们可以看到,在 NewGgzx 类中,我们可以传入一个实现 ICourse 接口的课程类,我们在Test类中调用的时候,只需要传入一个课程类即可调用学习方法,这样当想扩展新的内容,只需要创建一个新的课程类实现 ICourse 即可

Test使用

NewGgzx newGgzx =new NewGgzx();

newGgzx.studyMoocCourse(new WebCourse());

newGgzx.studyMoocCourse(new com.ggzx.design.priciple.dependenceiversion.JavaCourse());

从面向实现到面向接口,我们处理问题的方法改变了:

开始时,我们需要考虑在Test类中调用Ggzx里面的哪一种学习方法,即注重调用什么方法能够实现特定的课程到面向接口编程,我们考虑传入什么课程即可实现学习

当业务需求拓展时,拓展方法也改变了:

面向实现:需要改变底层的代码来协调我们需要使用的功能,用上面的例子来解释就是:当你想要学习一个课程,你就需要改变你底层的实现,增加新的代码面向接口:想学习什么课程,不会对其他课程造成影响,也不会影响到低层的Ggzx 。实际操作就是增加一门新的课程即可,实现接口之后,传入这个类到Ggzx的方法中就可以学习这一门课了

相对于细节的多变性,抽象的东西更稳定,以抽象为基础搭建的架构比以细节搭建的架构更加稳定

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Tomcat多实例与负载均衡
下一篇:【云原生 | Devops篇】深入Devops(一)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~