你好,你的問題包含了兩個(gè)小問題,①java設(shè)計(jì)interface的原因?②是不是為了盡可能替代類的繼承?
接口與繼承分別是什么?(定義)接口是一系列方法的聲明,比如方法名、參數(shù)、返回值等信息,接口中的方法不實(shí)現(xiàn),這些方法可以在不同的地方被不同的類實(shí)現(xiàn)。
繼承就是子類繼承父類的特征和行為,使得子類具有父類的實(shí)例域和方法。
接口與繼承的設(shè)計(jì)原因是什么?(用處)接口的主要作用在于降低代碼的耦合度,屏蔽實(shí)現(xiàn)層,比如前后端接口交互的時(shí)候,大家約定好接口層就可以互不影響的干活了,至于接口實(shí)現(xiàn)后端可以慢慢做。
繼承的主要作用在于,在已有基礎(chǔ)上繼續(xù)進(jìn)行功能的擴(kuò)充①清晰體現(xiàn)相關(guān)類間的層次結(jié)構(gòu)關(guān)系②減小代碼的冗余度,大大增加程序的重用性。
接口與繼承有什么區(qū)別?①定義的修飾符不同(interface),(extends)
②接口中只能定義全局常量和抽象方法,而在繼承中可以定義屬性方法,變量,常量等。
③接口被類實(shí)現(xiàn)時(shí),在類中一定要實(shí)現(xiàn)接口中的所有方法,而繼承可以調(diào)用指定方法。
④繼承只能繼承一個(gè)類,但implements可以實(shí)現(xiàn)多個(gè)接口,用逗號分開就行了 。
綜上所述,java中接口與類繼承各有自己存在的原因,有自己的適用場合,有區(qū)別也有一定的聯(lián)系,可以根據(jù)自己的具體需求來選擇。
繼承代價(jià)太大,強(qiáng)制你把父類的所有東西都拿過來,不然就不準(zhǔn)使用,這種將方法和屬性強(qiáng)耦合的方式容易導(dǎo)致類繼承多了到后面積重難返。使用鴨子類型(接口)能避免一部分繼承的缺陷,go里甚至直接繼承都沒有了,只有接口。
以前我們把物體抽象,大象抽象是動(dòng)物,動(dòng)物抽象是生物,這樣本身就把對象的內(nèi)容和行為綁定了,比如大象會(huì)噴水,就給大象加個(gè)噴水的方法,但是如果魚也會(huì)噴水,噴水這個(gè)方法又不能放到動(dòng)物類里,那么基于大象和魚的噴水方法實(shí)現(xiàn)的上層方法就無法復(fù)用,這兩個(gè)噴水在編譯器看來是沒有任何關(guān)系的。
開始人們想到搞一個(gè)噴水動(dòng)物類,繼承動(dòng)物類,大象和魚再繼承噴水動(dòng)物類。但是這樣終究治標(biāo)不治本,再有其他的變化,繼承鏈又要修改。
因?yàn)閷?shí)際上我們使用對象,都是在使用其方法(屬性其實(shí)也應(yīng)該算方法)。
我對面向?qū)ο蟪绦蜻^程的理解,這個(gè)過程實(shí)際上就是讓對象之間使用方法互發(fā)消息進(jìn)行通信和動(dòng)作,最終完成工作。
大家都知道的一句話,上層應(yīng)該依賴抽象而不是依賴細(xì)節(jié),然而依賴一個(gè)基類,本身已經(jīng)依賴這個(gè)基類的實(shí)現(xiàn)細(xì)節(jié),基類要求有個(gè)int成員,那么任何子類無論如何都需要有個(gè)int成員。那么理應(yīng)將方法抽象出來,而不去關(guān)心其到底是什么,因?yàn)槲覀儾⒉皇褂脤ο髢?nèi)部的內(nèi)容,我們只使用方法。
比如有個(gè)iwriteable接口,表示對象可以按字節(jié)寫入,那么上層的代碼就不用管寫的到底是什么了,只要能寫就行,就可以基于這個(gè)iwriteable接口里的方法,寫出例如寫字符串,寫圖片等方法。任何實(shí)現(xiàn)了這個(gè)接口的類,都可以復(fù)用這些寫字符串,寫圖片的代碼。
那么就有人要問了,那如果我確實(shí)要求要有個(gè)int成員,因?yàn)槲疑蠈哟a要使用呢?那么根據(jù)前面說的,屬性也是方法,你應(yīng)該再定義一個(gè)接口,比如這個(gè)int存的是年齡,那么就來個(gè)haveage接口并實(shí)現(xiàn)它。調(diào)用的地方可以要求對象實(shí)現(xiàn)哪幾個(gè)接口,這樣也能獲取需要的屬性。
這么一來,連屬性都沒有了,那么繼承也可以沒有了,畢竟你要求的不是基類了,是一個(gè)或多個(gè)接口的組合了,所以你可以看到,接口替代繼承是很自然的,而不是什么刻意而為的,是更高級抽象的體現(xiàn)。事實(shí)上基于自然規(guī)則的那套大象是動(dòng)物,動(dòng)物是生物的那套面向?qū)ο笠?guī)則,在程序設(shè)計(jì)里并不好用。所以如果讓我介紹面向?qū)ο螅也粫?huì)講這些例子。
這個(gè)問題很有意思。雖然有一些場景,使用繼承和接口都可以實(shí)現(xiàn),但是接口的存在絕對不僅僅是代替類的繼承。
首先繼承和接口的區(qū)別很明顯,用通俗的話來講,接口好比一個(gè)人的老師,會(huì)告訴你要做什么(實(shí)現(xiàn)接口要實(shí)現(xiàn)接口中的方法),而繼承就好比是親爹,會(huì)把所有的都給你(子類可以調(diào)用父類提供的方法),一個(gè)人可以有多個(gè)老師,但是只有一個(gè)親爹(接口可以實(shí)現(xiàn)多個(gè),類只可以繼承一個(gè))。
那么再來說說接口存在的意義。簡單的理解在Java中接口相當(dāng)于是定義了規(guī)范,而這些規(guī)范可以嚴(yán)格控制每個(gè)實(shí)現(xiàn)的功能。最明顯的應(yīng)用就是JavaEE,JavaEE中只定義了各種接口,并沒有實(shí)現(xiàn),而我們平時(shí)所使用的基本都是一些實(shí)現(xiàn)了這些接口的第三方類,比如tomcat的。
另外接口的存在也使得應(yīng)用的可維護(hù)性和擴(kuò)展性變得更強(qiáng),比如,在一個(gè)應(yīng)用中使用了MySQL數(shù)據(jù)庫,然后未來某一時(shí)間想要更換成其他數(shù)據(jù)庫,那么就只需要學(xué)一個(gè)其他數(shù)據(jù)庫的類實(shí)現(xiàn)數(shù)據(jù)庫接口就可以無縫切換了。
此外還有一個(gè)很重要的點(diǎn),接口是Java程序中解耦的重要手段。相反的類的繼承是確實(shí)增加了耦合度。
所以說,Java中接口的存在是很有必要的。