1. 程式人生 > 程式設計 >Java this super程式碼例項及使用方法總結

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也有三種用法:

1.普通的直接引用

與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。

3.引用建構函式

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關鍵字。

一、this

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;
  }
}

執行結果如下:

-----------
未知
空
101
-----------
遊客
空
101

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

第一、通過this呼叫另一個構造方法,用法是this(引數列表),這個僅僅在類的構造方法中,別的地方不能這麼用。

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

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

二、super

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的用法也不過這些,只有理解了其中的原理,才不會跌入陷阱!

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。