天天看點

Java中this和super的用法總結

這幾天看到類在繼承時會用到this和super,這裡就做了一點總結,與各位共同交流,有錯誤請各位指正~

一、this

this是自身的一個對象,代表對象本身,可以了解為:指向對象本身的一個指針。

this的用法在java中大體可以分為3種:

1.普通的直接引用

這種就不用講了,this相當于是指向目前對象本身。

2.形參與成員名字重名,用this來區分:

package com.demo;

public class Person {
    
    private int age = 10;
    
    public Person(){
        System.out.println("初始化年齡:"+age);
    }
    
    public int GetAge(int age){
        this.age = age;
        return this.age;
    }
}      
package com.demo;

public class Test1 {
    
    public static void main(String[] args) {
        Person Harry = new Person();
        System.out.println("Harry's age is "+Harry.GetAge(12));
    }
}      

運作結果:

初始化年齡:10
Harry's age is 12      

3.引用構造函數

這個和super放在一起講,見下面。

二、super

super可以了解為是指向自己超(父)類對象的一個指針,而這個超類指的是離自己最近的一個父類。

super也有三種用法:

與this類似,super相當于是指向目前對象的父類,這樣就可以用super.xxx來引用父類的成員。

2.子類中的成員變量或方法與父類中的成員變量或方法同名

package com.demo;

public class Country {

    String name;
    void value() {
       name = "China";
    }
}      
package com.demo;

public class City extends Country{

    String name;

    void value() {
        name = "Shanghai";
        super.value(); // 調用父類的方法
        System.out.println(name);
        System.out.println(super.name);
    }

    public static void main(String[] args) {
        City c = new City();
        c.value();
    }
}      
Shanghai
China      

可以看到,這裡既調用了父類的方法,也調用了父類的變量。若不調用父類方法value(),隻調用父類變量name的話,則父類name值為預設值null。

super(參數):調用父類中的某一個構造函數(應該為構造函數中的第一條語句)。

this(參數):調用本類中另一種形式的構造函數(應該為構造函數中的第一條語句)。

package com.demo.test;

public class Person {

    public static void prt(String s) {
        System.out.println(s);
    }

    Person() {
        prt("父類·無參數構造方法: " + "A Person.");
    }// 構造方法(1)

    Person(String name) {
        prt("父類·含一個參數的構造方法: " + "A person's name is " + name);
    }// 構造方法(2)
}      
package com.demo.test;

public class Chinese extends Person {

    Chinese() {
        super(); // 調用父類構造方法(1)
        prt("子類·調用父類“無參數構造方法“: " + "A chinese coder.");
    }

    Chinese(String name) {
        super(name);// 調用父類具有相同形參的構造方法(2)
        prt("子類·調用父類”含一個參數的構造方法“: " + "his name is " + name);
    }

    Chinese(String name, int age) {
        this(name);// 調用具有相同形參的構造方法(3)
        prt("子類:調用子類具有相同形參的構造方法:his age is " + age);
    }

    public static void main(String[] args) {
        Chinese cn = new Chinese();
        cn = new Chinese("codersai");
        cn = new Chinese("codersai", 18);
    }
}      
父類·無參數構造方法: A Person.
子類·調用父類“無參數構造方法“: A chinese coder.
父類·含一個參數的構造方法: A person's name is codersai
子類·調用父類”含一個參數的構造方法“: his name is codersai
父類·含一個參數的構造方法: A person's name is codersai
子類·調用父類”含一個參數的構造方法“: his name is codersai
子類:調用子類具有相同形參的構造方法:his age is 18      

從本例可以看到,可以用super和this分别調用父類的構造方法和本類中其他形式的構造方法。

例子中Chinese類第三種構造方法調用的是本類中第二種構造方法,而第二種構造方法是調用父類的,是以也要先調用父類的構造方法,再調用本類中第二種,最後是重寫第三種構造方法。

三、super 和 this 的異同

  • super(參數):調用基類中的某一個構造函數(應該為構造函數中的第一條語句) 
  • this(參數):調用本類中另一種形成的構造函數(應該為構造函數中的第一條語句)
  • super:它引用目前對象的直接父類中的成員(用來通路直接父類中被隐藏的父類中成員資料或函數,基類與派生類中有相同成員定義時如:super.變量名    super.成員函資料名(實參)
  • this:它代表目前對象名(在程式中易産生二義性之處,應使用this來指明目前對象;如果函數的形參與類中的成員資料同名,這時需用this來指明成員變量名)
  • 調用super()必須寫在子類構造方法的第一行,否則編譯不通過。每個子類構造方法的第一條語句,都是隐含地調用super(),如果父類沒有這種形式的構造函數,那麼在編譯的時候就會報錯。
  • super()和this()類似,差別是,super()從子類中調用父類的構造方法,this()在同一類内調用其它方法。
  • super()和this()均需放在構造方法内第一行。
  • 盡管可以用this調用一個構造器,但卻不能調用兩個。
  • this和super不能同時出現在一個構造函數裡面,因為this必然會調用其它的構造函數,其它的構造函數必然也會有super語句的存在,是以在同一個構造函數裡面有相同的語句,就失去了語句的意義,編譯器也不會通過。
  • this()和super()都指的是對象,是以,均不可以在static環境中使用。包括:static變量,static方法,static語句塊。
  • 從本質上講,this是一個指向本對象的指針, 然而super是一個Java關鍵字。

      Java關鍵字this隻能用于方法體内。當一個對象建立後,Java虛拟機(JVM)就會給這個對象配置設定一個引用自身的指針,這個指針的名字就是this。是以,this隻能在類中的非靜态方法中使用,靜态方法和靜态的代碼塊中絕對不能出現this。并且this隻和特定的對象關聯,而不和類關聯,同一個類的不同對象有不同的this。下面給出一個使用this的綜合執行個體,以便說明問題:

package com.demo.test;

public class Testthis {

    private int number;
    private String username;
    private String password;
    private int x = 100;

    public Testthis(int n) {
        number = n; // 這個還可以寫為: this.number=n;
    }

    public Testthis(int i, String username, String password) {
        // 成員變量和參數同名,成員變量被屏蔽,用"this.成員變量"的方式通路成員變量.
        this.username = username;
        this.password = password;
    }

    // 預設不帶參數的構造方法
    public Testthis() {
        this(0, "未知", "空"); // 通過this調用另一個構造方法
    }

    public Testthis(String name) {
        this(1, name, "空"); // 通過this調用另一個構造方法
    }

    public static void main(String args[]) {
        Testthis t1 = new Testthis();
        Testthis t2 = new Testthis("遊客");
        t1.outinfo(t1);
        t2.outinfo(t2);
    }

    private void outinfo(Testthis t) {
        System.out.println("-----------");
        System.out.println(t.number);
        System.out.println(t.username);
        System.out.println(t.password);
        f(); // 這個可以寫為: this.f();
    }

    private void f() {
        // 局部變量與成員變量同名,成員變量被屏蔽,用"this.成員變量"的方式通路成員變量.
        int x;
        x = this.x++;
        System.out.println(x);
        System.out.println(this.x);
    }

    // 傳回目前執行個體的引用
    private Testthis getSelf() {
        return this;
    }
}      

運作結果如下:

-----------
0
未知
空
100
101
-----------
0
遊客
空
100
101      

看着上面的例子,說明在什麼情況下需要用到this:

       第一、通過this調用另一個構造方法,用法是this(參數清單),這個僅僅在類的構造方法中,别的地方不能這麼用。

       第二、函數參數或者函數中的局部變量和成員變量同名的情況下,成員變量被屏蔽,此時要通路成員變量則需要用“this.成員變量名”的方式來引用成員變量。當然,在沒有同名的情況下,可以直接用成員變量的名字,而不用this,用了也不為錯,呵呵。

       第三、在函數中,需要引用該函數所屬類的目前對象的時候,直接用this。其實這些用法總結都是從對“this是指向對象本身的一個指針”這句話的更深入的了解而來的,死記不然容易忘記而且容易搞錯,要了解!

      super關鍵字和this作用類似,使被屏蔽的成員變量或者成員方法變為可見,或者說用來引用被屏蔽的成員變量和成員方法。不過super是用在子類中,目的是通路直接父類中被屏蔽的成員,注意是直接父類(就是類之上最近的超類)。下面是一個綜合運用super的例子,有兩個類:一個Father類,一個Father類的子類Son,通過這兩個類完全示範了super的用法,以下是代碼:

package com.demo.test;

public class Father {

    public String v = "Father";
    public String x = "輸出了Father類的public成員變量x!!!";

    public Father() {
        System.out.println("Father構造方法被調用!");
    }

    public Father(String v) {
        this.v = "Father類的帶參數構造方法!運作了.";
    }

    public void outinfo() {
        System.out.println("Father的outinfo方法被調用");
    }

    public static void main(String[] args) {
        // TODO 自動生成方法存根
    }
}      
package com.demo.test;

public class Son extends Father {

    public String v = "Son";

    public Son() {
        super(); // 調用超類的構造方法,隻能放到第一行.
        System.out.println("Son無參數構造方法被調用!");
        // super(); //錯誤的,必須放到構造方法體的最前面.
    }

    public Son(String str) {
        super(str);
        System.out.println("Son帶參數構造方法被調用!");
    }

    // 覆寫了超類成員方法outinfo()
    public void outinfo() {
        System.out.println("Son的outinfo()方法被調用");
    }

    public void test() {

        String v = "哈哈哈哈!"; // 局部變量v覆寫了成員變量v和超類變量v

        System.out.println("------1-----");
        System.out.println(v); // 輸出局部變量v
        System.out.println(this.v); // 輸出(子類)成員變量v
        System.out.println(super.v); // 輸出超類成員變量v

        System.out.println("------2-----");
        System.out.println(x); // 輸出超類成員變量v,子類繼承而來
        System.out.println(super.x); // 輸出超類成員變量v

        System.out.println("------3-----");
        outinfo(); // 調用子類的outinfo()方法
        this.outinfo(); // 調用子類的outinfo()方法
        super.outinfo(); // 調用父類的outinfo()方法
    }

    public static void main(String[] args) {
        new Son().test();

    }
}      
Father構造方法被調用!
Son無參數構造方法被調用!
------1-----
哈哈哈哈!
Son
Father
------2-----
輸出了Father類的public成員變量x!!!
輸出了Father類的public成員變量x!!!
------3-----
Son的outinfo()方法被調用
Son的outinfo()方法被調用
Father的outinfo方法被調用      

說明:此例子僅僅為了說明super的用法,實際在設計類的時候一般都盡可能私有(private)化。

通過上面的例子,下面總結一下super的用法:

      第一、在子類構造方法中要調用父類的構造方法,用“super(參數清單)”的方式調用,參數不是必須的。同時還要注意的一點是:“super(參數清單)”這條語句隻能用在子類構造方法體中的第一行。

      第二、當子類方法中的局部變量或者子類的成員變量與父類成員變量同名時,也就是子類局部變量覆寫父類成員變量時,用“super.成員變量名”來引用父類成員變量。當然,如果父類的成員變量沒有被覆寫,也可以用“super.成員變量名”來引用父類成員變量,不過這是不必要的。

      第三、當子類的成員方法覆寫了父類的成員方法時,也就是子類和父類有完全相同的方法定義(但方法體可以不同),此時,用“super.方法名(參數清單)”的方式通路父類的方法。

this、super的用法也不過這些,隻有了解了其中的原理,才不會跌入陷阱!