java的接口解耦方式
java的接口解耦方式
我只想把抽象的東西說的具體,或者說,聽起來簡單些,明白些。。。
學過java的人都知道,java是單繼承的,也就是說一個class只能繼承一個類。
例如我們想制作一臺有播放器的手機,那么我們先得制作一個播放器吧,再把播放器放進手機里。在java會怎么實現(xiàn)呢?如果使用繼承,我們會先創(chuàng)建一個播放器類,播放器類里面含有播放歌曲功能(方法),創(chuàng)建一個手機類繼承播放器類,重寫播放器的播歌功能(不重寫的話,會直接使用播放器自己定制的播放功能),這樣,我們就可以使用手機的播歌功能了。
現(xiàn)在,我們想制作一臺既有播歌功能,又有收音機功能的手機,那么我們該怎么辦?難道我們又要讓繼承了播放器的手機再繼承收音機?但是java是單繼承的,行不通,這時,接口應運而生!
接口,乍一看就是包含幾個方法的一個東西,它里面不包含具體實現(xiàn)的代碼,只包含方法的返回類型,名稱,參數(shù)列表,它代表的是一個功能的集合,只要實現(xiàn)了這個接口的類,他就具有了這些功能。
回到之前說的既有播歌又有收音機功能的手機,現(xiàn)在我們不把播放器和收音機封裝成類了,我們把它們封裝成接口(接口就是功能的集合),創(chuàng)建手機類,實現(xiàn)播放器和收音機接口,這樣看起來,是不是有點像多繼承?這違背了java的單繼承原則嗎?
其實沒有,有些書里面提到的多重繼承指的是多個實現(xiàn)接口。繼承(extends),是一種 is-a 關(guān)系的,所謂is-a關(guān)系,就是類似于“手機是播放器”或者“手機是收音機”這樣說法,但明顯,我們不能說“手機既是播放器又是收音機”,那么“他究竟是播放器還是收音機?”,這時大家就會可能這樣發(fā)問了,這就是java是單繼承的原因。實現(xiàn)(implement),是一種hava-a關(guān)系的,所謂have-a就是具有“某一項功能“的意思,我們這時候會說“手機既有播放器功能,又有收音機功能”,這樣的表達該明白了吧!讓手機再添加其他功能,只要再讓他實現(xiàn)那些功能接口就好了。
好像說了那么多還沒說到正題~哈哈,其實舉前面的例子我是想說明一個問題:只要一個方法操作的是類而非接口,那么你只能使用這個類及其之類。如果你想要將這個方法應用于不在此繼承結(jié)構(gòu)中的某個類,那么你就觸霉頭了。接口可以在很大程度上放寬這種限制,因此,他使我們可以編寫可復用性更好的代碼?!谩秚hinking in java》的某一些話。
舉例子
我需要一個鬧鐘,放在我床邊,每天叫我起床。但是我家里沒鬧鐘,只有一臺有鬧鐘功能的手機和一臺有鬧鐘功能的洗衣機。我需要的只是鬧鐘功能,我管他是什么,只要他能讓我起床就好了。如果某一天我連手機都丟了,我能把洗衣機放在我床邊叫我起床嗎?當然可以,因為洗衣機實現(xiàn)了鬧鐘功能。所以,我們經(jīng)常會這么做:把“鬧鐘”這個功能(而不是具體的某一項事物,如手機或者洗衣機)放在床邊,如果我們想聽洗衣機的鬧鐘聲就擺洗衣機,如果想聽手機的鬧鐘聲就擺手機。
從上面的例子,我們傳遞的不是某個具體的對象,而是一個抽象的“鬧鐘功能”的概念,至于實際上傳遞的是什么參數(shù),要看具體情況(取決于我想聽哪一種鬧鐘聲)。實際上,我們只關(guān)心“具有鬧鐘功能”這件事,我們不關(guān)心它是由誰實現(xiàn)的和怎樣實現(xiàn)的,這就做到了“請求”和“實現(xiàn)”分離開來,這就是接口的解耦!
java接口解耦效果的理解
先看一段代碼
public class A { public void say() { System.out.println("I am A"); } } public class C { public void put(A a) { a.say(); } }
在C類里面想調(diào)用一個含有say功能的東西,就把A傳入了,突然有一天,產(chǎn)品經(jīng)理多了個需求,想讓B這種類型也能在C中作為參數(shù)調(diào)用B的say,也即A,B這兩種類型都能滿足作為輸入,
public class B { public void say() { System.out.println("I am B"); } }
那好煩,不僅要像上面一樣新建B類,還要修改C的代碼適配需求,例如改成下面這個樣子,搞成重載的樣子,萬一產(chǎn)品經(jīng)理又來要求把D,E,F...這些具有say功能的類可以當成參數(shù)輸入,新建D,E,F這些類也就算了,難免嘛,問題是還要修改C的類,在里面再添加很多類似的修改,感覺耦合的很緊,代碼寫的太死板了,變動下需求就得改主程序C里面的代碼,
public class C{ public void put(A a) { a.say(); } public void put(B b) { b.say(); } }
有沒有其他的好辦法?有,用接口的方法,例如下面
public interface IBase { void say(); } public class A implements IBase { @Override public void say() { System.out.println("I am A"); } } public class B implements IBase { @Override public void say() { System.out.println("I am B"); } } public class C { public void put(IBase base) { base.say(); } }
定義一個接口,IBase,讓產(chǎn)品經(jīng)理新加的需求B,D,E,F都實現(xiàn)這個類,并且把C類里的參數(shù)寫成接口的形式(只要實現(xiàn)了我這個接口功能的都能傳入),這樣只要是實現(xiàn)了IBase接口的類,也即實現(xiàn)具備了實現(xiàn)接口say功能的任何類都可以傳進來,所以以后只需要新建B,D,E,F時實現(xiàn)這個接口就行了,不需要在C類里面修改源代碼。
這就大大降低了工作量,本質(zhì)上就是降低了耦合度,體現(xiàn)了接口的解耦效果,這是一點關(guān)于對接口解耦作用的理解
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java實現(xiàn)連接kubernates集群的兩種方式詳解
這篇文章主要為大家詳細介紹了Java實現(xiàn)連接kubernates集群的兩種方式,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-01-01SpringBoot+layuimini實現(xiàn)左側(cè)菜單動態(tài)展示的示例代碼
Layuimini是Layui的升級版,它是專業(yè)做后臺頁面的框架,而且是適合PC端和移動端,以下地址可以在PC端顯示,也可以在手機上顯示,只不過會做自適應,本文將給大家介紹了SpringBoot+layuimini實現(xiàn)左側(cè)菜單動態(tài)展示的方法,需要的朋友可以參考下2024-04-04SpringBoot與spring security的結(jié)合的示例
這篇文章主要介紹了SpringBoot與spring security的結(jié)合的示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-03-03IntelliJ IDEA中出現(xiàn)"PSI and index do not match"錯誤的解決辦法
今天小編就為大家分享一篇關(guān)于IntelliJ IDEA中出現(xiàn)"PSI and index do not match"錯誤的解決辦法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-10-10Java報錯:Error:java:?程序包org.springframework.boot不存在解決辦法
建完springboot項目時,點擊啟動,有可能會報錯,下面這篇文章主要給大家介紹了關(guān)于Java報錯:Error:java:?程序包org.springframework.boot不存在的解決辦法,需要的朋友可以參考下2024-02-02