溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java:接口和抽象類,傻傻分不清楚?

發(fā)布時(shí)間:2020-05-04 10:12:05 來(lái)源:網(wǎng)絡(luò) 閱讀:236 作者:沉默王二 欄目:編程語(yǔ)言

再來(lái)聊聊接口和抽象類。

01、抽象類和接口的區(qū)別

來(lái)看網(wǎng)絡(luò)上對(duì)接口的一番解釋:

接口(英文:Interface),在 Java 編程語(yǔ)言中是一個(gè)抽象類型,是抽象方法的集合。一個(gè)類通過(guò)繼承接口的方式,從而來(lái)繼承接口的抽象方法。

兄弟們,你們?cè)趺纯?,這段解釋把我繞得暈乎乎的,好像喝過(guò)一斤二鍋頭。到底是解釋抽象類呢還是接口呢?傻傻分不清楚。

搞不清楚要用抽象類還是接口,就先來(lái)看看兩者之間的區(qū)別。來(lái),抽象類和接口,你倆過(guò)來(lái)比比身高。

  1. 抽象類中的方法可以有方法體,能實(shí)現(xiàn)方法具體要實(shí)現(xiàn)的功能,但是接口中的方法不行,沒(méi)有方法體。
  2. 抽象類中的成員變量可以是各種類型的,而接口中的成員變量只能是 public static final 類型的,并且是隱式的,缺省的。
  3. 接口中不能含有靜態(tài)代碼塊以及靜態(tài)方法(用 static 修飾的方法),而抽象類是可以有靜態(tài)代碼塊和靜態(tài)方法的。
  4. 一個(gè)類只能繼承一個(gè)抽象類,而一個(gè)類卻可以實(shí)現(xiàn)多個(gè)接口。

02、進(jìn)一步剖析接口和抽象類

好像知道了兩者之間的區(qū)別,但印象還是有些模糊。沒(méi)關(guān)系,我們進(jìn)一步深入。

抽象類

抽象類體現(xiàn)了數(shù)據(jù)抽象的思想(不然呢),是實(shí)現(xiàn)多態(tài)的一種機(jī)制。抽象類定義了一組抽象的方法,至于這組抽象方法的具體表現(xiàn)形式由子類來(lái)繼承實(shí)現(xiàn)。

抽象類就是用來(lái)繼承的,否則它就沒(méi)有存在的任何意義。舉個(gè)例子,我們來(lái)定義一個(gè)抽象的作者類。

abstract class Author {
    abstract void write ();

    public void sleep () {
        System.out.println("吃飯睡覺(jué)打豆豆");
    }
}

作為一名作者,本職工作就是搞寫(xiě)作的,其他時(shí)間就吃飯睡覺(jué)打豆豆;但至于能寫(xiě)出什么樣的作品,就要看是哪一個(gè)作者了。比如說(shuō),沉默王二能寫(xiě)出的作品一定是幽默風(fēng)趣的。

public class Wanger extends Author {

    @Override
    void write() {
        System.out.println("沉默王二的作品《Web 全棧開(kāi)發(fā)進(jìn)階之路》,讀起來(lái)輕松愜意");
    }

}

注意到了沒(méi)?抽象類是可以有自己的方法的,但繼承它的子類可以忽視。

接口

接口是一種比抽象類更加抽象的“類”,畢竟是用關(guān)鍵字 interface 聲明的,不是用 class。

接口只是一種形式,就好像一紙契約,自身不能做任何事情。但只要某個(gè)類實(shí)現(xiàn)了這個(gè)接口,就必須按照這紙契約來(lái)辦事:接口里提到的方法必須全部實(shí)現(xiàn),少一個(gè)都不行(抽象類的子類可以忽視非抽象方法)。舉個(gè)例子,我們來(lái)定義一個(gè)北航出版合同的接口。

interface ContractBeihang {
    void scriptBeihang();
}

一旦作者簽訂了合同,那么就必須定期完成一定量的書(shū)稿。

public class Wanger extends Author implements ContractBeihang {

    @Override
    void write() {
        System.out.println("作品《Web 全棧開(kāi)發(fā)進(jìn)階之路》,讀起來(lái)輕松愜意的技術(shù)書(shū)");
    }

    @Override
    public void scriptBeihang() {
        System.out.println("一年內(nèi)完成書(shū)稿啊,不然要交違約金的哦。");
    }

}

接口是抽象類的補(bǔ)充,Java 為了保證數(shù)據(jù)的安全性不允許多重繼承,也就是說(shuō)一個(gè)類同時(shí)只允許繼承一個(gè)父類(為什么呢?請(qǐng)搜索關(guān)鍵字“菱形問(wèn)題”)。

但是接口不同,一個(gè)類可以同時(shí)實(shí)現(xiàn)多個(gè)接口,這些接口之間可以沒(méi)有多大的關(guān)系(彌補(bǔ)了抽象類不能多重繼承的缺陷)。比如說(shuō),沉默王二不僅簽了北航出版社的合同,還和 51CTO 簽了付費(fèi)課程的合同。

public class Wanger extends Author implements ContractBeihang, Contract51 {

    @Override
    void write() {
        System.out.println("作品《Web 全棧開(kāi)發(fā)進(jìn)階之路》,讀起來(lái)輕松愜意的技術(shù)書(shū)");
    }

    @Override
    public void scriptBeihang() {
        System.out.println("一年內(nèi)完成書(shū)稿啊,不然要交違約金的哦。");
    }

    @Override
    public void script51() {
        System.out.println("王老師,先把 Java 云盤(pán)的大綱整理出來(lái)。");
    }

}

03、接口和抽象類之間的差別

通過(guò)上面舉的例子,是不是對(duì)接口和抽象類有比較清晰的認(rèn)知了?如果還沒(méi)有,來(lái)來(lái)來(lái),我們?cè)賮?lái)比較一下接口和抽象類之間的差別。

Java:接口和抽象類,傻傻分不清楚?

究竟什么時(shí)候使用接口,什么時(shí)候使用抽象類呢?

1、抽象類表示了一種“is-a”的關(guān)系,而接口表示的是“l(fā)ike-a”的關(guān)系。也就是說(shuō),如果 B 類是 A(沉默王二是一個(gè)作者),則 A 應(yīng)該用抽象類。如果 B 類只是和 A 有某種關(guān)系,則 A 應(yīng)該用接口。

2、 如果要擁有自己的成員變量和非抽象方法,則用抽象類。接口只能存在靜態(tài)的不可變的成員變量(不過(guò)一般都不在接口中定義成員變量)。

3、為接口添加任何方法(抽象的),相應(yīng)的所有實(shí)現(xiàn)了這個(gè)接口的類,也必須實(shí)現(xiàn)新增的方法,否則會(huì)出現(xiàn)編譯錯(cuò)誤。對(duì)于抽象類,如果添加了非抽象方法,其子類卻可以坐享其成,完全不必?fù)?dān)心編譯會(huì)出問(wèn)題。

4、抽象類和接口有很大的相似性,請(qǐng)謹(jǐn)慎判斷。Java 從1.8版本開(kāi)始,嘗試向接口中引入了默認(rèn)方法和靜態(tài)方法,以此來(lái)減少抽象類和接口之間的差異。換句話說(shuō),兩者之間越來(lái)越難區(qū)分了。

04、接口的實(shí)際應(yīng)用

在實(shí)際的開(kāi)發(fā)應(yīng)用當(dāng)中,抽象類我用得不多(這可真是大實(shí)話);接口我倒是用得蠻多的,就像下面這樣子:

public interface CityMapper {

    @Select("select * from city")
    List<City> getCitys();

}

@Insert@Update、@Delete、@Select 被稱為 Mybatis 的注射器注解。

是不是突然感覺(jué)有點(diǎn)懵?之前還在談接口和抽象類,怎么一下子跳躍到 Mybatis 上面了呢?還有什么映射器注解?

嗯,這就對(duì)了。所有的理論知識(shí)都要應(yīng)用于實(shí)踐,否則也就沒(méi)有了存在價(jià)值。在我的實(shí)踐應(yīng)用當(dāng)中,接口用得最多的就是 MybatisMapper 接口。

MyBatis 是一款優(yōu)秀的持久層框架,它支持定制化 SQL、存儲(chǔ)過(guò)程以及高級(jí)映射。MyBatis 避免了幾乎所有的 JDBC 代碼和手動(dòng)設(shè)置參數(shù)以及獲取結(jié)果集。MyBatis 可以使用簡(jiǎn)單的 XML 或注解(就是你在前面見(jiàn)到的增刪改查四大注解)來(lái)配置和映射原生類型、接口和 Java 的 POJO(Plain Old Java Objects,普通老式 Java 對(duì)象)為數(shù)據(jù)庫(kù)中的記錄。

當(dāng)我們配置好了 MyBatis 環(huán)境后,可以直接通過(guò)以下語(yǔ)句來(lái)調(diào)用注射器接口。

@Service
public class CityService {
    @Autowired
    private CityMapper cityMapper;

    public void init() {
            List<City> citys = cityMapper.getCitys();
        }
    }
}

在注射器接口中,也只會(huì)存在那些與數(shù)據(jù)庫(kù)查詢相關(guān)的抽象方法,就像你看到的 List&lt;City&gt; getCitys();。一個(gè)注射器接口 + 注射器注解就可以增刪改查數(shù)據(jù)庫(kù),是不是感覺(jué)很神奇?

05、總結(jié)

這篇文章的目的是幫助更多的讀者了解和掌握抽象類、接口的特點(diǎn),以及不同的使用場(chǎng)景。

上一篇:再談 Java 的繼承和超類 Object

下一篇:Java 生成二維碼分享海報(bào)

謝謝大家的閱讀,原創(chuàng)不易,喜歡就隨手點(diǎn)個(gè)贊

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI