java學習-關於繼承以及構造方法、關鍵字super和this的一道題
題目 這段程序的輸出爲?
public class Test extends TT
{
public static void main(String args[])
{
Test t = new Test("Tom");
}
public Test(String s)
{
super(s);
System.out.println("How do you do?");
}
public Test()
{
this("I am Tom");
}
}
class TT
{
public TT()
{
System.out.println("What a pleasure!");
}
public TT(String s)
{
this();
System.out.println("I am "+s);
}
}
我的初步認知
因爲之前學java很久之前,看這段代碼時還帶着函數式編程的想法,認爲主函數裏只是new了一個對象,而沒有輸出,認爲不會有輸出。
進一步認識
認爲答案課本不過沒這麼簡單,便去請教了一位大佬,大佬痛心疾首的給我解釋了這段程序的輸出過程,讓我瞭解了自己對知識淺薄的認識。
解疑
從第一句我們瞭解到Test繼承的是父類TT
public class Test extends TT
1、Test在主類中new了一個對象,便結束了,此時就要了解構造方法是在實例化的時候自動執行
。 s
2、又因爲對象帶參數,執行帶參數的構造方法(Test (String s))。
public Test(String s)
{
super(s);
System.out.println("How do you do?");
}
3、此時到又因爲這個(Test (String s))中有super(s),此時又要知道super此時指向的是父類即TT,所以此時調用的是TT的構造函數,又因爲super帶參數,調用的方法爲
public TT(String s)
{
this();
System.out.println("I am "+s);
}
4、到這個方法中發現有this(),又要知道this指向的是當前對象,又因爲不帶參數,所以調用的是
public TT()
{
System.out.println("What a pleasure!");
}
5、回到Test中執行輸出 “How do you do?”
結果
What a pleasure!
I am Tom
How do you do?
知識梳理
1、實例化對象時會自動執行構造方法
2、super的用法
super可以理解爲是指向自己超(父)類對象的一個指針,而這個超類指的是離自己最近的一個父類。
(1)、普通的直接引用
與this類似,super相當於是指向當前對象的父類,這樣就可以用super.xxx來引用父類的成員。
(2)、子類中的成員變量或方法與父類中的成員變量或方法同名
class Country {
String name;
void value() {
name = "China";
}
}
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
(3)、引用構造函數
super(參數):調用父類中的某一個構造函數(應該爲構造函數中的第一條語句)。
this(參數):調用本類中另一種形式的構造函數(應該爲構造函數中的第一條語句)。
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)
}
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類第三種構造方法調用的是本類中第二種構造方法,而第二種構造方法是調用父類的,因此也要先調用父類的構造方法,再調用本類中第二種,最後是重寫第三種構造方法。
3、this的用法
this是自身的一個對象,代表對象本身,可以理解爲:指向對象本身的一個指針。
大致有三種用法:
(1)、普通的直接引用
this相當於是指向當前對象本身
(2)、形參與成員名字重名,用this來區分:
class Person {
private int age = 10;
public Person(){
System.out.println("初始化年齡:"+age);
}
public int GetAge(int age){
this.age = age;
return this.age;
}
}
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
可以看到,這裏age是GetAge成員方法的形參,this.age是Person類的成員變量。
(3)、引用構造函數
this(參數):調用本類中另一種形式的構造函數(應該爲構造函數中的第一條語句)。
例子在super的第三種用法中。
4、this和super的異同
(1)、super(參數):調用基類中的某一個構造函數(應該爲構造函數中的第一條語句)
(2)、this(參數):調用本類中另一種形成的構造函數(應該爲構造函數中的第一條語句)
(3)、super: 它引用當前對象的直接父類中的成員(用來訪問直接父類中被隱藏的父類中成員數據或函數,基類與派生類中有相同成員定義時如:super.變量名 super.成員函數據名(實參)
(4)、this:它代表當前對象名(在程序中易產生二義性之處,應使用this來指明當前對象;如果函數的形參與類中的成員數據同名,這時需用this來指明成員變量名)
(5)、調用super()必須寫在子類構造方法的第一行,否則編譯不通過。每個子類構造方法的第一條語句,都是隱含地調用super(),如果父類沒有這種形式的構造函數,那麼在編譯的時候就會報錯。
(6)、super()和this()類似,區別是,super()從子類中調用父類的構造方法,this()在同一類內調用其它方法。
(7)、super()和this()均需放在構造方法內第一行。
(8)、儘管可以用this調用一個構造器,但卻不能調用兩個。
(9)、this和super不能同時出現在一個構造函數裏面,因爲this必然會調用其它的構造函數,其它的構造函數必然也會有super語句的存在,所以在同一個構造函數裏面有相同的語句,就失去了語句的意義,編譯器也不會通過。
(10)、this()和super()都指的是對象,所以,均不可以在static環境中使用。包括:static變量,static方法,static語句塊。
(11)、從本質上講,this是一個指向本對象的指針, 然而super是一個Java關鍵字。