Java 你知道什么是耦合、如何解(降低)耦合
什么是耦合性
耦合性(Coupling),也叫耦合度,是對(duì)模塊間關(guān)聯(lián)程度的度量。模塊間的耦合度是指模塊之間的依賴(lài)關(guān)系,包括控制關(guān)系、調(diào)用關(guān)系、數(shù)據(jù)傳遞關(guān)系。模塊間聯(lián)系越多,其耦合性越強(qiáng),同時(shí)表明其獨(dú)立性越差( 降低耦合性,可以提高其獨(dú)立性)。
什么是程序間的耦合
假如:當(dāng)我去new一個(gè)對(duì)象的時(shí)候,而這個(gè)對(duì)象不存在,這個(gè)時(shí)候程序會(huì)報(bào)編譯時(shí)異常,也就意味著程序連運(yùn)行都運(yùn)行不了,我們可以理解為他們的耦合度較高。
如何解耦
我們可以利用Java的反射技術(shù),通過(guò)類(lèi)定名,來(lái)進(jìn)行反射創(chuàng)建對(duì)象,這個(gè)時(shí)候我們可以成功的避免編譯時(shí)異常,并且保證了項(xiàng)目在這個(gè)時(shí)候還能正常運(yùn)行。
工廠模式解耦
在實(shí)際開(kāi)發(fā)中我們可以把三層的對(duì)象都使用配置文件配置起來(lái),當(dāng)啟動(dòng)服務(wù)器應(yīng)用加載的時(shí)候,讓一個(gè)類(lèi)中的方法通過(guò)讀取配置文件,把這些對(duì)象創(chuàng)建出來(lái)并存起來(lái)。在接下來(lái)的使用的時(shí)候,直接拿過(guò)來(lái)用就好了。那么,這個(gè)讀取配置文件,創(chuàng)建和獲取三層對(duì)象的類(lèi)就是工廠。
案例
早期我們的 JDBC 操作,注冊(cè)驅(qū)動(dòng)時(shí),我們?yōu)槭裁床皇褂?DriverManager 的 register 方法,而是采用 Class.forName 的方式?
public class JdbcDemo1 { public static void main(String[] args) throws Exception { //1.注冊(cè)驅(qū)動(dòng) //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); //2.獲取連接 //3.獲取預(yù)處理 sql 語(yǔ)句對(duì)象 //4.獲取結(jié)果集 //5.遍歷結(jié)果集 } }
原因就是:
我們的類(lèi)依賴(lài)了數(shù)據(jù)庫(kù)的具體驅(qū)動(dòng)類(lèi)(MySQL),如果這時(shí)候更換了數(shù)據(jù)庫(kù)品牌(比如 Oracle),需要修改源碼來(lái)重新數(shù)據(jù)庫(kù)驅(qū)動(dòng)。這顯然不是我們想要的。
解決思路:
當(dāng)是我們講解 jdbc 時(shí),是通過(guò)反射來(lái)注冊(cè)驅(qū)動(dòng)的,代碼如下:
Class.forName("com.mysql.jdbc.Driver");//此處只是一個(gè)字符串
此時(shí)的好處是,我們的類(lèi)中不再依賴(lài)具體的驅(qū)動(dòng)類(lèi),此時(shí)就算刪除 mysql 的驅(qū)動(dòng) jar 包,依然可以編譯(運(yùn)行就不要想了,沒(méi)有驅(qū)動(dòng)不可能運(yùn)行成功的)。同時(shí),也產(chǎn)生了一個(gè)新的問(wèn)題,mysql 驅(qū)動(dòng)的全限定類(lèi)名字符串是在 java 類(lèi)中寫(xiě)死的,一旦要改還是要修改源碼。解決這個(gè)問(wèn)題也很簡(jiǎn)單,使用配置文件配置。
以上只是說(shuō)的解耦的一種思維,方便大家理解。
解耦的核心思想:若要減少代碼的耦合度,一定要盡量做到各個(gè)功能的代碼不能交叉編寫(xiě),編寫(xiě)封閉的代碼。
解耦合
廣大程序猿同胞,經(jīng)常會(huì)看到“解耦合”,也有很多人,會(huì)用這個(gè)詞來(lái)裝X,但是,實(shí)際真正能理解的人,并不多。接下來(lái),帶大家深入淺出的走一遍,如何解耦合。
首先,我們要知道,為什么要解耦合:通常,我們做一個(gè)項(xiàng)目,會(huì)用到很多基礎(chǔ)功能塊,比如xxx通信協(xié)議,xxxView等等,我們會(huì)把這種功能塊封裝成一個(gè)庫(kù),如果這個(gè)庫(kù),只能在這個(gè)指定的項(xiàng)目運(yùn)行,這就叫高耦合,這就導(dǎo)致了,如果下次再次遇到一個(gè)類(lèi)似的項(xiàng)目,需要用到同樣功能的功能塊時(shí),你會(huì)要做很多重復(fù)工作。假設(shè),每次使用json時(shí),你都要對(duì)json庫(kù)進(jìn)行改造,那將會(huì)是一個(gè)晴天霹靂。
但是,事與愿違,有些情況,還真的不太好解耦。
這里,我們先舉個(gè)栗子,比如排序
一個(gè)排序功能,對(duì)于大部分比較初級(jí)的程序猿來(lái)說(shuō),可能會(huì)寫(xiě)成這樣:
sort(List<Integer> list)
這樣就導(dǎo)致了一個(gè)問(wèn)題所在,這個(gè)方法只能排序int型數(shù)據(jù),如果下一個(gè)項(xiàng)目,需要用到對(duì)String進(jìn)行排序,那就很尷尬,感覺(jué)明明要成功了,但是又差一點(diǎn)。對(duì),就是差這一點(diǎn),就是代碼解耦的關(guān)鍵。
我們先要明確,我們需要做的是排序功能,在這個(gè)過(guò)程中,我們不可避免的需要使用2個(gè)數(shù)據(jù)的大小對(duì)比,而這個(gè)數(shù)據(jù),可能是任何數(shù)據(jù),也就是說(shuō),排序算法,我們是可以確定下來(lái),做成不動(dòng)的庫(kù),但是有一個(gè)數(shù)據(jù)大小匹配是我們無(wú)法做到的,或者說(shuō)是庫(kù)的耦合點(diǎn),那怎么辦呢?
我們就讓使用我們這個(gè)功能塊的人,告訴我們就行啦。
下面,我們參考Android庫(kù)里面,有個(gè)排序的api
Collections.sort(List<T> list, Comparator<? super T> c);
這里,Comparator這個(gè)接口,就是使用者,需要實(shí)現(xiàn),并且傳遞進(jìn)去的接口。這樣做,這個(gè)排序功能塊就可以應(yīng)用在任何場(chǎng)合,達(dá)到一次開(kāi)發(fā),受用終身的目的。是不是很神奇?
我們?cè)倥e個(gè)栗子,socket
我們?cè)陂_(kāi)發(fā)時(shí),經(jīng)常會(huì)用到socket庫(kù),而socket最常用,最常用的一個(gè)功能就是:“連接->發(fā)送數(shù)據(jù)->接收數(shù)據(jù)->斷開(kāi)連接->回調(diào)結(jié)果”
所以,如果需要把這個(gè)流程,封裝成一個(gè)功能塊是很有意義的。
但是,這里有一個(gè)問(wèn)題,是阻礙封裝的,就是 "接收數(shù)據(jù)->斷開(kāi)連接",socket讀取數(shù)據(jù)時(shí),是一個(gè)inputStream,是個(gè)流,也就是說(shuō),其實(shí),你并不知道,數(shù)據(jù)怎么樣才算接收 完整/完畢
可能,有的協(xié)議,是通過(guò)頭2個(gè)字節(jié)來(lái)判斷整個(gè)數(shù)據(jù)長(zhǎng)度
可能,有的協(xié)議是有幀頭,幀尾,轉(zhuǎn)義符來(lái)判斷整個(gè)數(shù)據(jù)長(zhǎng)度
……
這讓我們很頭疼,那怎么 解決了,既然無(wú)法知道的東西,就讓?xiě)?yīng)用程序來(lái)告訴你唄。和上面一樣,傳入一個(gè)協(xié)議實(shí)現(xiàn)唄:
public interface UnZipDataAction{ // 返回null,表示未接收完全,繼續(xù)接收,返回完整的byte[]就認(rèn)為是已經(jīng)接收完畢,把結(jié)果返回給應(yīng)用,并且斷開(kāi)連接 byte[] getRealData(byte[] recvData); ErrorCode getErrorCode(); }
這樣,我們就把“連接->發(fā)送數(shù)據(jù)->接收數(shù)據(jù)->斷開(kāi)連接->回調(diào)結(jié)果”整個(gè)流程封裝成了通用的功能塊了。
解耦總結(jié)來(lái)說(shuō)就是:你能知道的東西就寫(xiě)死,不知道但是又必須知道的東西,就讓?xiě)?yīng)用程序來(lái)告訴你,在java里面叫接口,在有些語(yǔ)言(OC, swift, C/C++)里面叫做代碼段。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
eclipse構(gòu)建和發(fā)布maven項(xiàng)目的教程
這篇文章主要為大家詳細(xì)介紹了eclipse構(gòu)建和發(fā)布maven項(xiàng)目的教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03關(guān)于Arrays.sort()使用的注意事項(xiàng)
這篇文章主要介紹了關(guān)于Arrays.sort()使用的注意事項(xiàng),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05vue 實(shí)現(xiàn)刪除對(duì)象的元素 delete
這篇文章主要介紹了vue 實(shí)現(xiàn)刪除對(duì)象的元素delete,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03mybatis plus開(kāi)發(fā)過(guò)程中遇到的問(wèn)題記錄及解決
這篇文章主要介紹了mybatis plus開(kāi)發(fā)過(guò)程中遇到的問(wèn)題記錄及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07Spring Boot jpa Service層代碼實(shí)例
這篇文章主要介紹了Spring Boot jpa Service層代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10Java使用組合模式實(shí)現(xiàn)表示公司組織結(jié)構(gòu)功能示例
這篇文章主要介紹了Java使用組合模式實(shí)現(xiàn)表示公司組織結(jié)構(gòu)功能,簡(jiǎn)單描述了組合模式的概念、功能并結(jié)合實(shí)例形式分析了Java使用組合模式實(shí)現(xiàn)公司組織結(jié)構(gòu)表示功能具體操作步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2018-05-05Java進(jìn)階學(xué)習(xí):jar打包詳解
Java進(jìn)階學(xué)習(xí):jar打包詳解...2006-12-12