溫馨提示×

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

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

this與super怎么在java項(xiàng)目中使用

發(fā)布時(shí)間:2021-03-20 15:10:45 來(lái)源:億速云 閱讀:135 作者:Leah 欄目:編程語(yǔ)言

this與super怎么在java項(xiàng)目中使用?針對(duì)這個(gè)問(wèn)題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問(wèn)題的小伙伴找到更簡(jiǎn)單易行的方法。

一、this

this是自身的一個(gè)對(duì)象,代表對(duì)象本身,可以理解為:指向?qū)ο蟊旧淼囊粋€(gè)指針。

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

1.普通的直接引用

這種就不用講了,this相當(dāng)于是指向當(dāng)前對(duì)象本身。

2.形參與成員名字重名,用this來(lái)區(qū)分:

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

運(yùn)行結(jié)果:

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

3.引用構(gòu)造函數(shù)

這個(gè)和super放在一起講,見(jiàn)下面。

二、super

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

super也有三種用法:

1.普通的直接引用

與this類似,super相當(dāng)于是指向當(dāng)前對(duì)象的父類,這樣就可以用super.xxx來(lái)引用父類的成員。

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(); // 調(diào)用父類的方法
    System.out.println(name);
    System.out.println(super.name);
  }

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

運(yùn)行結(jié)果:

Shanghai
China

可以看到,這里既調(diào)用了父類的方法,也調(diào)用了父類的變量。若不調(diào)用父類方法value(),只調(diào)用父類變量name的話,則父類name值為默認(rèn)值null。

3.引用構(gòu)造函數(shù)

super(參數(shù)):調(diào)用父類中的某一個(gè)構(gòu)造函數(shù)(應(yīng)該為構(gòu)造函數(shù)中的第一條語(yǔ)句)。

this(參數(shù)):調(diào)用本類中另一種形式的構(gòu)造函數(shù)(應(yīng)該為構(gòu)造函數(shù)中的第一條語(yǔ)句)。

package com.demo.test;

public class Person {

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

  Person() {
    prt("父類·無(wú)參數(shù)構(gòu)造方法: " + "A Person.");
  }// 構(gòu)造方法(1)

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

public class Chinese extends Person {

  Chinese() {
    super(); // 調(diào)用父類構(gòu)造方法(1)
    prt("子類·調(diào)用父類“無(wú)參數(shù)構(gòu)造方法“: " + "A chinese coder.");
  }

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

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

  public static void main(String[] args) {
    Chinese cn = new Chinese();
    cn = new Chinese("codersai");
    cn = new Chinese("codersai", 18);
  }
}

運(yùn)行結(jié)果:

父類·無(wú)參數(shù)構(gòu)造方法: A Person.
子類·調(diào)用父類“無(wú)參數(shù)構(gòu)造方法“: A chinese coder.
父類·含一個(gè)參數(shù)的構(gòu)造方法: A person's name is codersai
子類·調(diào)用父類”含一個(gè)參數(shù)的構(gòu)造方法“: his name is codersai
父類·含一個(gè)參數(shù)的構(gòu)造方法: A person's name is codersai
子類·調(diào)用父類”含一個(gè)參數(shù)的構(gòu)造方法“: his name is codersai
子類:調(diào)用子類具有相同形參的構(gòu)造方法:his age is 18

從本例可以看到,可以用super和this分別調(diào)用父類的構(gòu)造方法和本類中其他形式的構(gòu)造方法。

例子中Chinese類第三種構(gòu)造方法調(diào)用的是本類中第二種構(gòu)造方法,而第二種構(gòu)造方法是調(diào)用父類的,因此也要先調(diào)用父類的構(gòu)造方法,再調(diào)用本類中第二種,最后是重寫(xiě)第三種構(gòu)造方法。

三、super 和 this 的異同

  • super(參數(shù)):調(diào)用基類中的某一個(gè)構(gòu)造函數(shù)(應(yīng)該為構(gòu)造函數(shù)中的第一條語(yǔ)句)

  • this(參數(shù)):調(diào)用本類中另一種形成的構(gòu)造函數(shù)(應(yīng)該為構(gòu)造函數(shù)中的第一條語(yǔ)句)

  • super:它引用當(dāng)前對(duì)象的直接父類中的成員(用來(lái)訪問(wèn)直接父類中被隱藏的父類中成員數(shù)據(jù)或函數(shù),基類與派生類中有相同成員定義時(shí)如:super.變量名 super.成員函數(shù)據(jù)名(實(shí)參)

  • this:它代表當(dāng)前對(duì)象名(在程序中易產(chǎn)生二義性之處,應(yīng)使用this來(lái)指明當(dāng)前對(duì)象;如果函數(shù)的形參與類中的成員數(shù)據(jù)同名,這時(shí)需用this來(lái)指明成員變量名)

  • 調(diào)用super()必須寫(xiě)在子類構(gòu)造方法的第一行,否則編譯不通過(guò)。每個(gè)子類構(gòu)造方法的第一條語(yǔ)句,都是隱含地調(diào)用super(),如果父類沒(méi)有這種形式的構(gòu)造函數(shù),那么在編譯的時(shí)候就會(huì)報(bào)錯(cuò)。

  • super()和this()類似,區(qū)別是,super()從子類中調(diào)用父類的構(gòu)造方法,this()在同一類內(nèi)調(diào)用其它方法。

  • super()和this()均需放在構(gòu)造方法內(nèi)第一行。

  • 盡管可以用this調(diào)用一個(gè)構(gòu)造器,但卻不能調(diào)用兩個(gè)。

  • this和super不能同時(shí)出現(xiàn)在一個(gè)構(gòu)造函數(shù)里面,因?yàn)閠his必然會(huì)調(diào)用其它的構(gòu)造函數(shù),其它的構(gòu)造函數(shù)必然也會(huì)有super語(yǔ)句的存在,所以在同一個(gè)構(gòu)造函數(shù)里面有相同的語(yǔ)句,就失去了語(yǔ)句的意義,編譯器也不會(huì)通過(guò)。

  • this()和super()都指的是對(duì)象,所以,均不可以在static環(huán)境中使用。包括:static變量,static方法,static語(yǔ)句塊。

  • 從本質(zhì)上講,this是一個(gè)指向本對(duì)象的指針, 然而super是一個(gè)Java關(guān)鍵字。

一、this

Java關(guān)鍵字this只能用于方法體內(nèi)。當(dāng)一個(gè)對(duì)象創(chuàng)建后,Java虛擬機(jī)(JVM)就會(huì)給這個(gè)對(duì)象分配一個(gè)引用自身的指針,這個(gè)指針的名字就是this。因此,this只能在類中的非靜態(tài)方法中使用,靜態(tài)方法和靜態(tài)的代碼塊中絕對(duì)不能出現(xiàn)this。并且this只和特定的對(duì)象關(guān)聯(lián),而不和類關(guān)聯(lián),同一個(gè)類的不同對(duì)象有不同的this。下面給出一個(gè)使用this的綜合實(shí)例,以便說(shuō)明問(wèn)題:

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; // 這個(gè)還可以寫(xiě)為: this.number=n;
  }

  public Testthis(int i, String username, String password) {
    // 成員變量和參數(shù)同名,成員變量被屏蔽,用"this.成員變量"的方式訪問(wèn)成員變量.
    this.username = username;
    this.password = password;
  }

  // 默認(rèn)不帶參數(shù)的構(gòu)造方法
  public Testthis() {
    this(0, "未知", "空"); // 通過(guò)this調(diào)用另一個(gè)構(gòu)造方法
  }

  public Testthis(String name) {
    this(1, name, "空"); // 通過(guò)this調(diào)用另一個(gè)構(gòu)造方法
  }

  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(); // 這個(gè)可以寫(xiě)為: this.f();
  }

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

  // 返回當(dāng)前實(shí)例的引用
  private Testthis getSelf() {
    return this;
  }
}

運(yùn)行結(jié)果如下:

-----------
未知
空
101
-----------
游客
空
101

看著上面的例子,說(shuō)明在什么情況下需要用到this:

第一、通過(guò)this調(diào)用另一個(gè)構(gòu)造方法,用法是this(參數(shù)列表),這個(gè)僅僅在類的構(gòu)造方法中,別的地方不能這么用。

第二、函數(shù)參數(shù)或者函數(shù)中的局部變量和成員變量同名的情況下,成員變量被屏蔽,此時(shí)要訪問(wèn)成員變量則需要用“this.成員變量名”的方式來(lái)引用成員變量。當(dāng)然,在沒(méi)有同名的情況下,可以直接用成員變量的名字,而不用this,用了也不為錯(cuò),呵呵。

第三、在函數(shù)中,需要引用該函數(shù)所屬類的當(dāng)前對(duì)象的時(shí)候,直接用this。其實(shí)這些用法總結(jié)都是從對(duì)“this是指向?qū)ο蟊旧淼囊粋€(gè)指針”這句話的更深入的理解而來(lái)的,死記不然容易忘記而且容易搞錯(cuò),要理解!

二、super

super關(guān)鍵字和this作用類似,使被屏蔽的成員變量或者成員方法變?yōu)榭梢?jiàn),或者說(shuō)用來(lái)引用被屏蔽的成員變量和成員方法。不過(guò)super是用在子類中,目的是訪問(wèn)直接父類中被屏蔽的成員,注意是直接父類(就是類之上最近的超類)。下面是一個(gè)綜合運(yùn)用super的例子,有兩個(gè)類:一個(gè)Father類,一個(gè)Father類的子類Son,通過(guò)這兩個(gè)類完全演示了super的用法,以下是代碼:

package com.demo.test;

public class Father {

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

  public Father() {
    System.out.println("Father構(gòu)造方法被調(diào)用!");
  }

  public Father(String v) {
    this.v = "Father類的帶參數(shù)構(gòu)造方法!運(yùn)行了.";
  }

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

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

public class Son extends Father {

  public String v = "Son";

  public Son() {
    super(); // 調(diào)用超類的構(gòu)造方法,只能放到第一行.
    System.out.println("Son無(wú)參數(shù)構(gòu)造方法被調(diào)用!");
    // super(); //錯(cuò)誤的,必須放到構(gòu)造方法體的最前面.
  }

  public Son(String str) {
    super(str);
    System.out.println("Son帶參數(shù)構(gòu)造方法被調(diào)用!");
  }

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

  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,子類繼承而來(lái)
    System.out.println(super.x); // 輸出超類成員變量v

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

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

  }
}

運(yùn)行結(jié)果:

Father構(gòu)造方法被調(diào)用!
Son無(wú)參數(shù)構(gòu)造方法被調(diào)用!
------1-----
哈哈哈哈!
Son
Father
------2-----
輸出了Father類的public成員變量x!!!
輸出了Father類的public成員變量x!!!
------3-----
Son的outinfo()方法被調(diào)用
Son的outinfo()方法被調(diào)用
Father的outinfo方法被調(diào)用

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

通過(guò)上面的例子,下面總結(jié)一下super的用法:

第一、在子類構(gòu)造方法中要調(diào)用父類的構(gòu)造方法,用“super(參數(shù)列表)”的方式調(diào)用,參數(shù)不是必須的。同時(shí)還要注意的一點(diǎn)是:“super(參數(shù)列表)”這條語(yǔ)句只能用在子類構(gòu)造方法體中的第一行。

第二、當(dāng)子類方法中的局部變量或者子類的成員變量與父類成員變量同名時(shí),也就是子類局部變量覆蓋父類成員變量時(shí),用“super.成員變量名”來(lái)引用父類成員變量。當(dāng)然,如果父類的成員變量沒(méi)有被覆蓋,也可以用“super.成員變量名”來(lái)引用父類成員變量,不過(guò)這是不必要的。

第三、當(dāng)子類的成員方法覆蓋了父類的成員方法時(shí),也就是子類和父類有完全相同的方法定義(但方法體可以不同),此時(shí),用“super.方法名(參數(shù)列表)”的方式訪問(wèn)父類的方法。

this、super的用法也不過(guò)這些,只有理解了其中的原理,才不會(huì)跌入陷阱!

關(guān)于this與super怎么在java項(xiàng)目中使用問(wèn)題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒(méi)有解開(kāi),可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識(shí)。

向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