Java設(shè)計(jì)模式之依賴(lài)倒轉(zhuǎn)原則精解
1.什么是依賴(lài)倒轉(zhuǎn)原則?
- 高層模塊不應(yīng)該依賴(lài)低層模塊,二者都應(yīng)該依賴(lài)其抽象。
- 抽象不應(yīng)該依賴(lài)細(xì)節(jié),細(xì)節(jié)應(yīng)該依賴(lài)抽象。
- 依賴(lài)倒轉(zhuǎn) (倒置) 的中心思想是面向接口編程。
- 依賴(lài)倒轉(zhuǎn)原則是基于這樣的設(shè)計(jì)理念:相對(duì)于細(xì)節(jié)的多變性,抽象的東西要穩(wěn)定的多。以抽象為基礎(chǔ)搭建的架構(gòu)比以細(xì)節(jié)為基礎(chǔ)的架構(gòu)要穩(wěn)定的多。在Java中,抽象指的是接口或抽象類(lèi),細(xì)節(jié)就是具體的實(shí)現(xiàn)類(lèi)。
- 使用接口或抽象類(lèi)的目的是制定好規(guī)范,而不涉及任何具體的操作,把展現(xiàn)細(xì)節(jié)的任務(wù)交給他們的實(shí)現(xiàn)類(lèi)去完成。
2.代碼案例
package com.szh.principle.inversion;
/**
*
*/
class Email {
public String getInfo() {
return "電子郵件信息: hello,world";
}
}
// 完成Person接收消息的功能
// 方式1分析
// 1. 簡(jiǎn)單,比較容易想到
// 2. 如果我們獲取的對(duì)象是 微信,短信等等,則新增類(lèi),同時(shí)Person也要增加相應(yīng)的接收方法
// 3. 解決思路:引入一個(gè)抽象的接口IReceiver, 表示接收者, 這樣Person類(lèi)與接口IReceiver發(fā)生依賴(lài)
// 因?yàn)镋mail, WeiXin 等等屬于接收的范圍,他們各自實(shí)現(xiàn)IReceiver 接口就ok, 這樣我們就符號(hào)依賴(lài)倒轉(zhuǎn)原則
class Person {
public void receive(Email email ) {
System.out.println(email.getInfo());
}
}
public class DependencyInversion {
public static void main(String[] args) {
Person person = new Person();
person.receive(new Email());
}
}

我們可以根據(jù)依賴(lài)倒轉(zhuǎn)原則對(duì)上面的代碼做一個(gè)改進(jìn),如下:??????
package com.szh.principle.inversion.improve;
/**
*
*/
//定義接口
interface IReceiver {
public String getInfo();
}
class Email implements IReceiver {
public String getInfo() {
return "電子郵件信息: hello,world";
}
}
//增加微信
class WeiXin implements IReceiver {
public String getInfo() {
return "微信信息: hello,ok";
}
}
//方式2
class Person {
//這里我們是對(duì)接口的依賴(lài)
public void receive(IReceiver receiver ) {
System.out.println(receiver.getInfo());
}
}
public class DependencyInversion {
public static void main(String[] args) {
//客戶(hù)端無(wú)需改變
Person person = new Person();
person.receive(new Email());
person.receive(new WeiXin());
}
}

3.依賴(lài)關(guān)系傳遞的三種方式和案例舉例
3.1 接口傳遞
package com.szh.principle.inversion.improve;
/**
* 方式1: 通過(guò)接口傳遞實(shí)現(xiàn)依賴(lài)
*/
interface IOpenAndClose1 {
public void open(ITV1 tv); //抽象方法,接收接口
}
interface ITV1 { //ITV接口
public void play();
}
class OpenAndClose1 implements IOpenAndClose1 {
@Override
public void open(ITV1 tv){
tv.play();
}
}
class ChangHong1 implements ITV1 {
@Override
public void play() {
System.out.println("長(zhǎng)虹電視機(jī),打開(kāi)");
}
}
public class DependencyPass1 {
public static void main(String[] args) {
ChangHong1 changHong = new ChangHong1();
OpenAndClose1 openAndClose = new OpenAndClose1();
openAndClose.open(changHong);
}
}

3.2 構(gòu)造方法傳遞
package com.szh.principle.inversion.improve;
/**
* 方式2: 通過(guò)構(gòu)造方法依賴(lài)傳遞
*/
interface IOpenAndClose2 {
public void open(); //抽象方法
}
interface ITV2 { //ITV接口
public void play();
}
class OpenAndClose2 implements IOpenAndClose2 {
public ITV2 tv; //成員
public OpenAndClose2(ITV2 tv){ //構(gòu)造器
this.tv = tv;
}
public void open(){
this.tv.play();
}
}
class ChangHong2 implements ITV2 {
public void play() {
System.out.println("長(zhǎng)虹電視機(jī),打開(kāi)");
}
}
public class DependencyPass2 {
public static void main(String[] args) {
ChangHong2 changHong = new ChangHong2();
//通過(guò)構(gòu)造器進(jìn)行依賴(lài)傳遞
OpenAndClose2 openAndClose = new OpenAndClose2(changHong);
openAndClose.open();
}
}

3.3 setter方法傳遞
package com.szh.principle.inversion.improve;
/**
* 方式3: 通過(guò)setter方法傳遞
*/
interface IOpenAndClose3 {
public void open(); // 抽象方法
public void setTv(ITV3 tv);
}
interface ITV3 { // ITV接口
public void play();
}
class OpenAndClose3 implements IOpenAndClose3 {
private ITV3 tv;
public void setTv(ITV3 tv) {
this.tv = tv;
}
public void open() {
this.tv.play();
}
}
class ChangHong3 implements ITV3 {
public void play() {
System.out.println("長(zhǎng)虹電視機(jī),打開(kāi)");
}
}
public class DependencyPass3 {
public static void main(String[] args) {
ChangHong3 changHong = new ChangHong3();
//通過(guò)setter方法進(jìn)行依賴(lài)傳遞
OpenAndClose3 openAndClose = new OpenAndClose3();
openAndClose.setTv(changHong);
openAndClose.open();
}
}

4.依賴(lài)倒轉(zhuǎn)原則總結(jié)
- 低層模塊盡量都要有抽象類(lèi)或接口,或者兩者都有,程序穩(wěn)定性更好。
- 變量的聲明類(lèi)型盡量是抽象類(lèi)或接口,這樣我們的變量引用和實(shí)際對(duì)象間,就存在一個(gè)緩沖層,利于程序擴(kuò)展和優(yōu)化。
- 繼承時(shí)遵循里氏替換原則。(我們后面再說(shuō))
到此這篇關(guān)于Java設(shè)計(jì)模式之依賴(lài)倒轉(zhuǎn)原則精解的文章就介紹到這了,更多相關(guān)Java 依賴(lài)倒轉(zhuǎn)原則內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 簡(jiǎn)單講解Java設(shè)計(jì)模式編程中的單一職責(zé)原則
- Java設(shè)計(jì)模式之接口隔離原則精解
- Java設(shè)計(jì)模式之里氏替換原則精解
- Java設(shè)計(jì)模式之開(kāi)閉原則精解
- Java設(shè)計(jì)模式之迪米特原則精解
- 詳解java設(shè)計(jì)模式中的門(mén)面模式
- JAVA設(shè)計(jì)模式之單例模式詳解
- java設(shè)計(jì)模式(實(shí)戰(zhàn))-責(zé)任鏈模式
- Java設(shè)計(jì)模式之職責(zé)鏈模式詳解
- Java設(shè)計(jì)模式之單一職責(zé)原則精解
相關(guān)文章
springboot+mybatis plus實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)查詢(xún)
實(shí)際開(kāi)發(fā)過(guò)程中經(jīng)常需要查詢(xún)節(jié)點(diǎn)樹(shù),根據(jù)指定節(jié)點(diǎn)獲取子節(jié)點(diǎn)列表,本文主要介紹了springboot+mybatis plus實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)查詢(xún),感興趣的可以了解一下2021-07-07
Mybatis使用foreach標(biāo)簽實(shí)現(xiàn)批量插入方式
這篇文章主要介紹了Mybatis使用foreach標(biāo)簽實(shí)現(xiàn)批量插入方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03
淺談Java之Map 按值排序 (Map sort by value)
下面小編就為大家?guī)?lái)一篇淺談Java之Map 按值排序 (Map sort by value)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08
Java集合類(lèi)知識(shí)點(diǎn)總結(jié)
本文把Java集合類(lèi)的相關(guān)知識(shí)點(diǎn)做了總結(jié),并把Java常用集合類(lèi)之間的區(qū)別做了分析,一起參考學(xué)習(xí)下。2018-02-02
數(shù)組在java中的擴(kuò)容的實(shí)例方法
在本篇文章里小編給大家分享的是一篇關(guān)于數(shù)組在java中的擴(kuò)容的實(shí)例方法內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。2021-01-01
Java SimpleDateFormat線(xiàn)程安全問(wèn)題原理詳解
這篇文章主要介紹了Java SimpleDateFormat線(xiàn)程安全問(wèn)題原理詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05
java實(shí)現(xiàn)jdbc批量插入數(shù)據(jù)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)jdbc批量插入數(shù)據(jù),三種JDBC批量插入編程方法進(jìn)行比較,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-05-05

