Java設(shè)計模式之依賴倒轉(zhuǎn)原則精解
1.什么是依賴倒轉(zhuǎn)原則?
- 高層模塊不應(yīng)該依賴低層模塊,二者都應(yīng)該依賴其抽象。
- 抽象不應(yīng)該依賴細(xì)節(jié),細(xì)節(jié)應(yīng)該依賴抽象。
- 依賴倒轉(zhuǎn) (倒置) 的中心思想是面向接口編程。
- 依賴倒轉(zhuǎn)原則是基于這樣的設(shè)計理念:相對于細(xì)節(jié)的多變性,抽象的東西要穩(wěn)定的多。以抽象為基礎(chǔ)搭建的架構(gòu)比以細(xì)節(jié)為基礎(chǔ)的架構(gòu)要穩(wěn)定的多。在Java中,抽象指的是接口或抽象類,細(xì)節(jié)就是具體的實(shí)現(xiàn)類。
- 使用接口或抽象類的目的是制定好規(guī)范,而不涉及任何具體的操作,把展現(xiàn)細(xì)節(jié)的任務(wù)交給他們的實(shí)現(xiàn)類去完成。
2.代碼案例
package com.szh.principle.inversion; /** * */ class Email { public String getInfo() { return "電子郵件信息: hello,world"; } } // 完成Person接收消息的功能 // 方式1分析 // 1. 簡單,比較容易想到 // 2. 如果我們獲取的對象是 微信,短信等等,則新增類,同時Person也要增加相應(yīng)的接收方法 // 3. 解決思路:引入一個抽象的接口IReceiver, 表示接收者, 這樣Person類與接口IReceiver發(fā)生依賴 // 因?yàn)镋mail, WeiXin 等等屬于接收的范圍,他們各自實(shí)現(xiàn)IReceiver 接口就ok, 這樣我們就符號依賴倒轉(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ù)依賴倒轉(zhuǎn)原則對上面的代碼做一個改進(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 { //這里我們是對接口的依賴 public void receive(IReceiver receiver ) { System.out.println(receiver.getInfo()); } } public class DependencyInversion { public static void main(String[] args) { //客戶端無需改變 Person person = new Person(); person.receive(new Email()); person.receive(new WeiXin()); } }
3.依賴關(guān)系傳遞的三種方式和案例舉例
3.1 接口傳遞
package com.szh.principle.inversion.improve; /** * 方式1: 通過接口傳遞實(shí)現(xiàn)依賴 */ 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("長虹電視機(jī),打開"); } } 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: 通過構(gòu)造方法依賴傳遞 */ 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("長虹電視機(jī),打開"); } } public class DependencyPass2 { public static void main(String[] args) { ChangHong2 changHong = new ChangHong2(); //通過構(gòu)造器進(jìn)行依賴傳遞 OpenAndClose2 openAndClose = new OpenAndClose2(changHong); openAndClose.open(); } }
3.3 setter方法傳遞
package com.szh.principle.inversion.improve; /** * 方式3: 通過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("長虹電視機(jī),打開"); } } public class DependencyPass3 { public static void main(String[] args) { ChangHong3 changHong = new ChangHong3(); //通過setter方法進(jìn)行依賴傳遞 OpenAndClose3 openAndClose = new OpenAndClose3(); openAndClose.setTv(changHong); openAndClose.open(); } }
4.依賴倒轉(zhuǎn)原則總結(jié)
- 低層模塊盡量都要有抽象類或接口,或者兩者都有,程序穩(wěn)定性更好。
- 變量的聲明類型盡量是抽象類或接口,這樣我們的變量引用和實(shí)際對象間,就存在一個緩沖層,利于程序擴(kuò)展和優(yōu)化。
- 繼承時遵循里氏替換原則。(我們后面再說)
到此這篇關(guān)于Java設(shè)計模式之依賴倒轉(zhuǎn)原則精解的文章就介紹到這了,更多相關(guān)Java 依賴倒轉(zhuǎn)原則內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot+mybatis plus實(shí)現(xiàn)樹形結(jié)構(gòu)查詢
實(shí)際開發(fā)過程中經(jīng)常需要查詢節(jié)點(diǎn)樹,根據(jù)指定節(jié)點(diǎn)獲取子節(jié)點(diǎn)列表,本文主要介紹了springboot+mybatis plus實(shí)現(xiàn)樹形結(jié)構(gòu)查詢,感興趣的可以了解一下2021-07-07Mybatis使用foreach標(biāo)簽實(shí)現(xiàn)批量插入方式
這篇文章主要介紹了Mybatis使用foreach標(biāo)簽實(shí)現(xiàn)批量插入方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-03-03淺談Java之Map 按值排序 (Map sort by value)
下面小編就為大家?guī)硪黄獪\談Java之Map 按值排序 (Map sort by value)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-08-08數(shù)組在java中的擴(kuò)容的實(shí)例方法
在本篇文章里小編給大家分享的是一篇關(guān)于數(shù)組在java中的擴(kuò)容的實(shí)例方法內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。2021-01-01Java SimpleDateFormat線程安全問題原理詳解
這篇文章主要介紹了Java SimpleDateFormat線程安全問題原理詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-05-05java實(shí)現(xiàn)jdbc批量插入數(shù)據(jù)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)jdbc批量插入數(shù)據(jù),三種JDBC批量插入編程方法進(jìn)行比較,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-05-05