您好,登錄后才能下訂單哦!
小編給大家分享一下Java中怎么解耦合,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
耦合性(Coupling),也叫耦合度,是對(duì)模塊間關(guān)聯(lián)程度的度量。模塊間的耦合度是指模塊之間的依賴關(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ù),通過類定名,來進(jìn)行反射創(chuàng)建對(duì)象,這個(gè)時(shí)候我們可以成功的避免編譯時(shí)異常,并且保證了項(xiàng)目在這個(gè)時(shí)候還能正常運(yùn)行。
在實(shí)際開發(fā)中我們可以把三層的對(duì)象都使用配置文件配置起來,當(dāng)啟動(dòng)服務(wù)器應(yīng)用加載的時(shí)候,讓一個(gè)類中的方法通過讀取配置文件,把這些對(duì)象創(chuàng)建出來并存起來。在接下來的使用的時(shí)候,直接拿過來用就好了。那么,這個(gè)讀取配置文件,創(chuàng)建和獲取三層對(duì)象的類就是工廠。
早期我們的 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 語句對(duì)象 //4.獲取結(jié)果集 //5.遍歷結(jié)果集 } }
我們的類依賴了數(shù)據(jù)庫的具體驅(qū)動(dòng)類(MySQL),如果這時(shí)候更換了數(shù)據(jù)庫品牌(比如 Oracle),需要修改源碼來重新數(shù)據(jù)庫驅(qū)動(dòng)。這顯然不是我們想要的。
當(dāng)是我們講解 jdbc 時(shí),是通過反射來注冊(cè)驅(qū)動(dòng)的,代碼如下:
Class.forName("com.mysql.jdbc.Driver");//此處只是一個(gè)字符串
此時(shí)的好處是,我們的類中不再依賴具體的驅(qū)動(dòng)類,此時(shí)就算刪除 mysql 的驅(qū)動(dòng) jar 包,依然可以編譯(運(yùn)行就不要想了,沒有驅(qū)動(dòng)不可能運(yùn)行成功的)。同時(shí),也產(chǎn)生了一個(gè)新的問題,mysql 驅(qū)動(dòng)的全限定類名字符串是在 java 類中寫死的,一旦要改還是要修改源碼。解決這個(gè)問題也很簡單,使用配置文件配置。
以上只是說的解耦的一種思維,方便大家理解。
解耦的核心思想:若要減少代碼的耦合度,一定要盡量做到各個(gè)功能的代碼不能交叉編寫,編寫封閉的代碼。
廣大程序猿同胞,經(jīng)常會(huì)看到“解耦合”,也有很多人,會(huì)用這個(gè)詞來裝X,但是,實(shí)際真正能理解的人,并不多。接下來,帶大家深入淺出的走一遍,如何解耦合。
首先,我們要知道,為什么要解耦合:通常,我們做一個(gè)項(xiàng)目,會(huì)用到很多基礎(chǔ)功能塊,比如xxx通信協(xié)議,xxxView等等,我們會(huì)把這種功能塊封裝成一個(gè)庫,如果這個(gè)庫,只能在這個(gè)指定的項(xiàng)目運(yùn)行,這就叫高耦合,這就導(dǎo)致了,如果下次再次遇到一個(gè)類似的項(xiàng)目,需要用到同樣功能的功能塊時(shí),你會(huì)要做很多重復(fù)工作。假設(shè),每次使用json時(shí),你都要對(duì)json庫進(jìn)行改造,那將會(huì)是一個(gè)晴天霹靂。
但是,事與愿違,有些情況,還真的不太好解耦。
一個(gè)排序功能,對(duì)于大部分比較初級(jí)的程序猿來說,可能會(huì)寫成這樣:
sort(List<Integer> list)
這樣就導(dǎo)致了一個(gè)問題所在,這個(gè)方法只能排序int型數(shù)據(jù),如果下一個(gè)項(xiàng)目,需要用到對(duì)String進(jìn)行排序,那就很尷尬,感覺明明要成功了,但是又差一點(diǎn)。對(duì),就是差這一點(diǎn),就是代碼解耦的關(guān)鍵。
我們先要明確,我們需要做的是排序功能,在這個(gè)過程中,我們不可避免的需要使用2個(gè)數(shù)據(jù)的大小對(duì)比,而這個(gè)數(shù)據(jù),可能是任何數(shù)據(jù),也就是說,排序算法,我們是可以確定下來,做成不動(dòng)的庫,但是有一個(gè)數(shù)據(jù)大小匹配是我們無法做到的,或者說是庫的耦合點(diǎn),那怎么辦呢?
我們就讓使用我們這個(gè)功能塊的人,告訴我們就行啦。
下面,我們參考Android庫里面,有個(gè)排序的api
Collections.sort(List<T> list, Comparator<? super T> c);
這里,Comparator這個(gè)接口,就是使用者,需要實(shí)現(xiàn),并且傳遞進(jìn)去的接口。這樣做,這個(gè)排序功能塊就可以應(yīng)用在任何場(chǎng)合,達(dá)到一次開發(fā),受用終身的目的。是不是很神奇?
我們?cè)陂_發(fā)時(shí),經(jīng)常會(huì)用到socket庫,而socket最常用,最常用的一個(gè)功能就是:“連接->發(fā)送數(shù)據(jù)->接收數(shù)據(jù)->斷開連接->回調(diào)結(jié)果”
所以,如果需要把這個(gè)流程,封裝成一個(gè)功能塊是很有意義的。
但是,這里有一個(gè)問題,是阻礙封裝的,就是 "接收數(shù)據(jù)->斷開連接",socket讀取數(shù)據(jù)時(shí),是一個(gè)inputStream,是個(gè)流,也就是說,其實(shí),你并不知道,數(shù)據(jù)怎么樣才算接收 完整/完畢
可能,有的協(xié)議,是通過頭2個(gè)字節(jié)來判斷整個(gè)數(shù)據(jù)長度
可能,有的協(xié)議是有幀頭,幀尾,轉(zhuǎn)義符來判斷整個(gè)數(shù)據(jù)長度
……
這讓我們很頭疼,那怎么 解決了,既然無法知道的東西,就讓應(yīng)用程序來告訴你唄。和上面一樣,傳入一個(gè)協(xié)議實(shí)現(xiàn)唄:
public interface UnZipDataAction{ // 返回null,表示未接收完全,繼續(xù)接收,返回完整的byte[]就認(rèn)為是已經(jīng)接收完畢,把結(jié)果返回給應(yīng)用,并且斷開連接 byte[] getRealData(byte[] recvData); ErrorCode getErrorCode(); }
這樣,我們就把“連接->發(fā)送數(shù)據(jù)->接收數(shù)據(jù)->斷開連接->回調(diào)結(jié)果”整個(gè)流程封裝成了通用的功能塊了。
解耦總結(jié)來說就是:你能知道的東西就寫死,不知道但是又必須知道的東西,就讓應(yīng)用程序來告訴你,在java里面叫接口,在有些語言(OC, swift, C/C++)里面叫做代碼段。
以上是“Java中怎么解耦合”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。