技術大佬:我去,你竟然還不會用 this 關鍵字

沉默王二發表於2020-05-24

上一篇文章寫的是 Spring Boot 的入門,結果有讀者留言說,Java 都還沒搞完,搞什麼 Spring Boot,唬得我一愣一愣的。那這篇就繼續來搞 Java,推出廣受好評的我去系列第四集:你竟然還不會用 this 關鍵字。

“老大,能給說詳細地說說 this 關鍵字嗎,總感覺對這個關鍵字的認知不夠全面。”小王又過來找我了,他問的態度很謙遜,很卑微,但我還是忍不住破口大罵:“我擦,小王,你丫的竟然不會用 this,我當初是怎麼面試你進來的!”

小王被我這句話嚇壞了,趕緊躲到自己崗位上改 bug 去了。我呢,加班加點開始寫這篇文章,真良心用苦啊。在 Java 中,this 關鍵字指的是當前物件(它的方法正在被呼叫)的引用,能理解吧,各位親?不理解的話,我們繼續往下看。

看完再不明白,你過來捶爆我,我保證不還手,只要不打臉。

01、消除欄位歧義

我敢賭一毛錢,所有的讀者,不管男女老少,應該都知道這種用法,畢竟寫構造方法的時候經常用啊。誰要不知道,過來,我給你發一毛錢紅包,只要你臉皮夠厚。

public class Writer {
    private int age;
    private String name;

    public Writer(int age, String name) {
        this.age = age;
        this.name = name;
    }
}

Writer 類有兩個成員變數,分別是 age 和 name,在使用有參建構函式的時候,如果引數名和成員變數的名字相同,就需要使用 this 關鍵字消除歧義:this.age 是指成員變數,age 是指構造方法的引數。

02、引用類的其他構造方法

當一個類的構造方法有多個,並且它們之間有交集的話,就可以使用 this 關鍵字來呼叫不同的構造方法,從而減少程式碼量。

比如說,在無參構造方法中呼叫有參構造方法:

public class Writer {
    private int age;
    private String name;

    public Writer(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public Writer() {
        this(18"沉默王二");
    }
}

也可以在有參構造方法中呼叫無參構造方法:

public class Writer {
    private int age;
    private String name;

    public Writer(int age, String name) {
        this();
        this.age = age;
        this.name = name;
    }

    public Writer() {
    }
}

需要注意的是,this() 必須是構造方法中的第一條語句,否則就會報錯。

03、作為引數傳遞

在下例中,有一個無參的構造方法,裡面呼叫了 print() 方法,引數只有一個 this 關鍵字。

public class ThisTest {
    public ThisTest() {
        print(this);
    }

    private void print(ThisTest thisTest) {
        System.out.println("print " +thisTest);
    }

    public static void main(String[] args) {
        ThisTest test = new ThisTest();
        System.out.println("main " + test);
    }
}

來列印看一下結果:

print com.cmower.baeldung.this1.ThisTest@573fd745
main com.cmower.baeldung.this1.ThisTest@573fd745

從結果中可以看得出來,this 就是我們在 main() 方法中使用 new 關鍵字建立的 ThisTest 物件。

04、鏈式呼叫

學過 JavaScript,或者 jQuery 的讀者可能對鏈式呼叫比較熟悉,類似於 a.b().c().d(),彷彿能無窮無盡呼叫下去。

在 Java 中,對應的專有名詞叫 Builder 模式,來看一個示例。

public class Writer {
    private int age;
    private String name;
    private String bookName;

    public Writer(WriterBuilder builder) {
        this.age = builder.age;
        this.name = builder.name;
        this.bookName = builder.bookName;
    }

    public static class WriterBuilder {
        public String bookName;
        private int age;
        private String name;

        public WriterBuilder(int age, String name) {
            this.age = age;
            this.name = name;
        }

        public WriterBuilder writeBook(String bookName) {
            this.bookName = bookName;
            return this;
        }

        public Writer build() {
            return new Writer(this);
        }
    }
}

Writer 類有三個成員變數,分別是 age、name 和 bookName,還有它們仨對應的一個構造方法,引數是一個內部靜態類 WriterBuilder。

內部類 WriterBuilder 也有三個成員變數,和 Writer 類一致,不同的是,WriterBuilder 類的構造方法裡面只有 age 和 name 賦值了,另外一個成員變數 bookName 通過單獨的方法 writeBook() 來賦值,注意,該方法的返回型別是 WriterBuilder,最後使用 return 返回了 this 關鍵字。

最後的 build() 方法用來建立一個 Writer 物件,引數為 this 關鍵字,也就是當前的 WriterBuilder 物件。

這時候,建立 Writer 物件就可以通過鏈式呼叫的方式。

Writer writer = new Writer.WriterBuilder(18,"沉默王二")
                .writeBook("《Web全棧開發進階之路》")
                .build();

05、在內部類中訪問外部類物件

說實話,自從 Java 8 的函數語言程式設計出現後,就很少用到 this 在內部類中訪問外部類物件了。來看一個示例:

public class ThisInnerTest {
    private String name;

    class InnerClass {
        public InnerClass() {
            ThisInnerTest thisInnerTest = ThisInnerTest.this;
            String outerName = thisInnerTest.name;
        }
    }
}

在內部類 InnerClass 的構造方法中,通過外部類.this 可以獲取到外部類物件,然後就可以使用外部類的成員變數了,比如說 name。

06、關於 super

本來想單獨寫一篇 super 關鍵字的,但可寫的內容不多。本質上,this 關鍵字和 super 關鍵字有蠻多相似之處的,所以,就放在 this 這篇文章的末尾說一說吧。

簡而言之,super 關鍵字就是用來訪問父類的。

先來看父類:

public class SuperBase {
    String message = "父類";

    public SuperBase(String message) {
        this.message = message;
    }

    public SuperBase() {
    }

    public void printMessage() {
        System.out.println(message);
    }
}

再來看子類:

public class SuperSub extends SuperBase {
    String message = "子類";

    public SuperSub(String message) {
        super(message);
    }

    public SuperSub() {
        super.printMessage();
        printMessage();
    }

    public void getParentMessage() {
        System.out.println(super.message);
    }

    public void printMessage() {
        System.out.println(message);
    }
}

1)super 關鍵字可用於訪問父類的構造方法

你看,子類可以通過 super(message) 來呼叫父類的構造方法。現在來新建一個 SuperSub 物件,看看輸出結果是什麼:

SuperSub superSub = new SuperSub("子類的message");

new 關鍵字在呼叫構造方法建立子類物件的時候,會通過 super 關鍵字初始化父類的 message,所以此此時父類的 message 會輸出“子類的message”。

2)super 關鍵字可以訪問父類的變數

上述例子中的 SuperSub 類中就有,getParentMessage() 通過 super.message 方法父類的同名成員變數 message。

3)當方法發生重寫時,super 關鍵字可以訪問父類的同名方法

上述例子中的 SuperSub 類中就有,無參的構造方法 SuperSub() 中就使用 super.printMessage() 呼叫了父類的同名方法。

07、總結

親愛的讀者朋友,我應該說得很全面了吧?我想小王看到了這篇文章後一定會感謝我的良苦用心的,他畢竟是個積極好學的好同事啊。

如果覺得文章對你有點幫助,請微信搜尋「 沉默王二 」第一時間閱讀,回覆「併發」更有一份阿里大牛重寫的 Java 併發程式設計實戰,從此再也不用擔心面試官在這方面的刁難了。

本文已收錄 GitHub,傳送門~ ,裡面更有大廠面試完整考點,歡迎 Star。

我是沉默王二,一枚有顏值卻靠才華苟且的程式設計師。關注即可提升學習效率,別忘了三連啊,點贊、收藏、留言,我不挑,嘻嘻

相關文章