1. 子類的構造函數如果要引用super的話,必須把super放在函數的首位
代碼如下:
class Base {
Base() {
System.out.println("Base");
}
}
public class Checket extends Base {
Checket() {
super();//調用父類的構造方法,一定要放在方法的首個語句
System.out.println("Checket");
}
public static void main(String argv[]) {
Checket c = new Checket();
}
}
如果想用super繼承父類構造的方法,但是沒有放在第一行的話,那么在super之前的語句,肯定是為了滿足自己想要完成某些行為的語句,但是又用了super繼承父類的構造方法。那么以前所做的修改就都回到以前了,就是說又成了父類的構造方法了。
2.在Java中,有時還會遇到子類中的成員變量或方法與超類(有時也稱父類)中的成員變量或方法同名。因為子類中的成員變量或方法名優先級高,所以子類中的同名成員變量或方法就隱藏了超類的成員變量或方法,但是我們如果想要使用超類中的這個成員變量或方法,就需要用到super.
代碼如下:
class Country {
String name;
void value() {
name = "China";
}
}
class City extends Country {
String name;
void value() {
name = "Hefei";
super.value();//不調用此方法時,super.name返回的是父類的成員變量的值null
System.out.println(name);
System.out.println(super.name);
}
public static void main(String[] args) {
City c=new City();
c.value();
}
}
為了在子類中引用父類中的成員變量name和方法value(),在代碼中使用了super、super.name和super.value(),若不調用super.value()時,super.name返回父類成員變量默認值null,調用此方法時,super.value()方法把成員變量name賦值為China,再利用super.name調用父類的成員變量的值。
另外,要注意的是super.name調用的是成員變量的值,
代碼如下:
class Country {
String name="xianfan";
String value(String name) {
name = "China";
return name;
}
}
class City extends Country {
String name;
String value(String name) {
name = "Hefei";
super.value("失敗");//不調用此方法時,super.name返回的是父類的成員變量的值null
System.out.println(name);
System.out.println(super.name);
return name;
}
public static void main(String[] args) {
City c=new City();
c.value("成功");
}
}
結果為:Hefei
xianfan
此時,super.name返回的值是父類成員變量的值xianfan,而此時的super.value()方法是不起作用的。
3.用super直接傳遞參數:
代碼如下:
class Person {
public static void prt(String s) {
System.out.println(s);
}
Person() {
prt("A Person.");
}
Person(String name) {
prt("A person name is:" + name);
}
}
public class Chinese extends Person {
Chinese() {
super(); // 調用父類構造函數(1)
prt("A chinese.");// (4)
}
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("kevin");
cn = new Chinese("kevin", 22);
}
}
結果為:A Person.
A chinese.
A person name is:kevin
his name is:kevin
A person name is:kevin
his name is:kevin
his age is:22
在這段程序中,this和super不再是像以前那樣用“.”連接一個方法或成員,而是直接在其后跟上適當的參數,因此它的意義也就有了變化。super后加參數的是用來調用父類中具有相同形式的構造函數,如1和2處。this后加參數則調用的是當前具有相同參數的構造函數,如3處。當然,在Chinese的各個重載構造函數中,this和super在一般方法中的各種用法也仍可使用,比如4處,你可以將它替換為“this.prt”(因為它繼承了父類中的那個方法)或者是“super.prt”(因為它是父類中的方法且可被子類訪問),它照樣可以正確運行。但這樣似乎就有點畫蛇添足的味道了。
4.super和this的異同:
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關鍵字。
this與super關鍵字在java中構造函數中的應用:
**
super()函數
**
super()函數在子類構造函數中調用父類的構造函數時使用,而且必須要在構造函數的第一行,例如:
classAnimal{
publicAnimal() {
System.out.println("An Animal");
}
}
classDogextendsAnimal{
publicDog() {
super();
System.out.println("A Dog");
//super();錯誤的,因為super()方法必須在構造函數的第一行
//如果子類構造函數中沒有寫super()函數,編譯器會自動幫我們添加一個無參數的super()
}
}
classTest{
publicstaticvoidmain(String [] args){
Dog dog =newDog();
}
}
執行這段代碼的結果為:
An Animal
A Dog
定義子類的一個對象時,會先調用子類的構造函數,然后在調用父類的構造函數,如果父類函數足夠多的話,會一直調用到最終的父類構造函數,函數調用時會使用棧空間,所以按照入棧的順序,最先進入的是子類的構造函數,然后才是鄰近的父類構造函數,最后再棧頂的是最終的父類構造函數,構造函數執行是則按照從棧頂到棧底的順序依次執行,所以本例中的執行結果是先執行Animal的構造函數,然后再執行子類的構造函數。
classAnimal{
privateString name;
publicString getName(){
name = name;
}
publicAnimal(String name) {
this.name = name;
}
}
classDogextendsAnimal{
publicDog() {
super(name);
}
}
classTest{
publicstaticvoidmain(String [] args){
Dog dog =newDog("jack");
System.out.println(dog.getName());
}
}
運行結果:
jack
當父類構造函數有參數時,如果要獲取父類的private的成員變量并給其賦值作為子類的結果,此時在定義子類的構造函數時就需要調用父類的構造函數,并傳值,如上例所示。
this()函數
this()函數主要應用于同一類中從某個構造函數調用另一個重載版的構造函數。this()只能用在構造函數中,并且也只能在第一行。所以在同一個構造函數中this()和super()不能同時出現。
例如下面的這個例子:
classMiniextendsCar{
Color color;
//無參數函數以默認的顏色調用真正的構造函數
publicMini() {
this(color.Red);
}
//真正的構造函數
publicMini(Color c){
super("mini");
color = c;
}
//不能同時調用super()和this(),因為他們只能選擇一個
publicMini(intsize) {
super(size);
this(color.Red);
}
·1
·2
·3
·4
·5
·6
·7
·8
·9
·10
·11
·12
·13
·14
·15
·16
`所以綜上所述,super()與this()的區別主要有以下:
不同點:
1、super()主要是對父類構造函數的調用,this()是對重載構造函數的調用
2、super()主要是在繼承了父類的子類的構造函數中使用,是在不同類中的使用;this()主要是在同一類的不同構造函數中的使用
相同點:
1、super()和this()都必須在構造函數的第一行進行調用,否則就是錯誤的