Java編程思想讀書筆記(6章)
發(fā)表時(shí)間:2024-05-21 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]第6章 重復(fù)運(yùn)用classes 一.繼承(inheritance) 1. 在derived class中overriding某個(gè)函數(shù)時(shí),只能覆寫base class中的接口,即base class中的public或protected或friendly函數(shù)。如果試圖overriding一個(gè)...
第6章 重復(fù)運(yùn)用classes
一.繼承(inheritance)
1. 在derived class中overriding某個(gè)函數(shù)時(shí),只能覆寫base class中的接口,即base class中的public或protected或friendly函數(shù)。如果試圖overriding一個(gè)private函數(shù),雖然編譯通過,但實(shí)際上你只是在derived class中添加了一個(gè)函數(shù)。如
class Cleanser{
private void prt(){//(b)
System.out.println("Cleanser.prt()");
}
}
public class ExplicitStatic extends Cleanser{
public void prt(){
System.out.println("ExplicitStatic.prt()");
}
public static void main(String[] args){
Cleanser x = new ExplicitStatic();
x.prt();//(a)
}
}
因?yàn)镃leanser中的prt()是private,所以不能在其derivedclass中被覆寫。ExplicitStatic中的prt()只是ExplicitStatic中的一個(gè)函數(shù),所以當(dāng)試圖在(a)處通過多態(tài)來調(diào)用prt()時(shí),會(huì)發(fā)生錯(cuò)誤。如果把(b)處的private去掉,則結(jié)果為
ExplicitStatic.prt()
2. Super的使用
1)通過關(guān)鍵字super可以調(diào)用當(dāng)前class的superclass(父類)。
例6.1.1.1
class Base{
Base(){System.out.println("Base()");}
public void scrub() { System.out.println(" Base.scrub()"); }
}
class Cleanser extends Base{
private String s = new String("Cleanser");
public void append(String a) { s+=a; }
public void dilute() { append(" dilute()"); }
public void apply() { append(" apply()"); }
public void scrub() { append(" scrub()"); }
public void print() { System.out.println(s); }
Cleanser(){
System.out.println("Cleanser(): " + s);
}
public static void testStatic(){
System.out.println("testStatic()");
}
public static void main(String[] args){
Cleanser x = new Cleanser();
x.dilute(); x.apply(); x.scrub(); x.print();
}
}
public class ExplicitStatic extends Cleanser{
ExplicitStatic(){
System.out.println("ExplicitStatic()");
}
public void scrub(){
append(" Detergen.scrub()");
super.testStatic();
super.scrub();//調(diào)用的是Cleanser.scrub()
}
public void foam() { append(" foam()"); }
public static void main(String[] args){
ExplicitStatic x = new ExplicitStatic();
x.dilute(); x.apply(); x.scrub(); x.foam();
x.print(); System.out.println("Test base class:");
Cleanser.main(args);
testStatic();
}
}
運(yùn)行結(jié)果:
Base()
Cleanser(): Cleanser
ExplicitStatic()
testStatic()
Cleanser dilute() apply() Detergen.scrub() scrub() foam()
Test base class:
Base()
Cleanser(): Cleanser
Cleanser dilute() apply() scrub()
testStatic()
2)通過super來調(diào)用superclass中的成員時(shí),調(diào)用的是最近成員。
例6.1.1.2
class Base{
protected String baseS = "Base";//(a)
//private String baseS = "Base";
Base(){System.out.println("Base()");}
}
class Cleanser extends Base{
protected String baseS = "Cleanser";//(b)
public String s = new String("Cleanser");
Cleanser(){
System.out.println("Cleanser(): " + s);
}
Cleanser(String a){
System.out.println("Cleanser(" + a + "): s = " + s );
}
}
public class ExplicitStatic extends Cleanser{
String s2 = s;
String baseS = super.baseS; //(c)
ExplicitStatic(){
super("ExplicitStatic");
System.out.println("ExplicitStatic():s2 = " + s2 + ", baseS = "
+ baseS + "super.baseS = " + super.baseS);
baseS = "ExplicitStatic";
System.out.println("baseS = " + baseS + " , super.baseS = " + super.baseS);
}
public static void main(String[] args){
ExplicitStatic x = new ExplicitStatic();
}
}
結(jié)果1:
Base()
Cleanser(ExplicitStatic): s = Cleanser
ExplicitStatic():s2 = Cleanser, baseS = Cleanser,super.baseS = Cleanser
baseS = ExplicitStatic , super.baseS = Cleanser
在上面例子中,在三個(gè)class中都存在String bases實(shí)例。在ExplicitStatic中如果直接調(diào)用baseS,則實(shí)際調(diào)用的是當(dāng)前類ExplicitStatic中的baseS(即(c)處的成員);如果通過super.bases來調(diào)用baseS,則調(diào)用的是離當(dāng)前類ExplicitStatic最近的baseS成員,即Cleanser class中的baseS實(shí)例(即(b)處),產(chǎn)生的結(jié)果如結(jié)果1所示。如果把(b)處語句注釋掉,則將調(diào)用Base class中的baseS,結(jié)果如結(jié)果2所示。
結(jié)果2:
Base()
Cleanser(ExplicitStatic): s = Cleanser
ExplicitStatic():s2 = Cleanser, baseS = Base,super.baseS = Base
baseS = ExplicitStatic , super.baseS = Base
3. Base class的初始化
2.1 當(dāng)你產(chǎn)生derived class對(duì)象時(shí),其中會(huì)包含base class子對(duì)象(subobject)。這個(gè)子對(duì)象就和你另外產(chǎn)生的base class對(duì)象一模一樣。
2.2 通過super()可調(diào)用base class的構(gòu)造函數(shù),但必須放在構(gòu)造函數(shù)的第一行,并且只能在構(gòu)造函數(shù)中運(yùn)用。
2.3 初始化順序?yàn)椋?
1) 加載代碼(.class文件)
2) 初始化class的靜態(tài)成員,初始化順序了“從里到外”,即從baseclass開始。
3) 在derived class的構(gòu)造函數(shù)中調(diào)用base class的構(gòu)造函數(shù)。
如果在derived class的構(gòu)造函數(shù)中沒有通過super()顯式調(diào)用調(diào)用base class的構(gòu)造函數(shù),編譯器會(huì)調(diào)用bass class的default構(gòu)造函數(shù)并自動(dòng)生成相應(yīng)的調(diào)用語句,從而產(chǎn)生一個(gè)base class實(shí)例。如果在derived class的構(gòu)造函數(shù)中通過super()顯示調(diào)用了父類的構(gòu)造函數(shù),則調(diào)用所指定的構(gòu)造函數(shù)。調(diào)用構(gòu)造函數(shù)的調(diào)用順序是“從里到外”。
4) 調(diào)用derived class的構(gòu)造函數(shù)。
**:當(dāng)base class沒有default構(gòu)造函數(shù)時(shí),必須在derived class的構(gòu)造函數(shù)中通過super顯示調(diào)用base class的構(gòu)造函數(shù)。
例:下面代碼的初始化過程為:
1) 裝載ExplicitStatic的代碼(裝載ExplicitStatic.class文件)。
2) 發(fā)現(xiàn)ExplicitStatic有關(guān)鍵字extends,裝載ExplicitStatic的baseclass的代碼(裝載Cleanser.class文件)。
3) 發(fā)現(xiàn)Cleanser有關(guān)鍵字extends,裝載Cleanser的baseclass的代碼(裝載Base.class文件)。
4) 初始化Base class中的靜態(tài)成員。
5) 初始化Cleanser class中的靜態(tài)成員。
6) 初始化ExplicitStatic class中的靜態(tài)成員。
如果把(c)處的代碼注釋掉,那么初始化工作到此就結(jié)束了。
7) 為ExplicitStatic對(duì)象分配存儲(chǔ)空間,并把存儲(chǔ)空間初始化為0。
8) 在ExplicitStatic class的構(gòu)造中調(diào)用super("ExplicitStatic")(在ExplicitStatic class的構(gòu)造函數(shù)中顯式調(diào)用父類的構(gòu)造函數(shù)),試圖產(chǎn)生一個(gè)Cleanser class實(shí)例。
9) 為Cleanser對(duì)象分配存儲(chǔ)空間,并把存儲(chǔ)空間初始化為0。
10) 由于Cleanser class又是繼承自Base class,會(huì)在Cleanser class的構(gòu)造函數(shù)中通過super()(由于沒有顯式調(diào)用父類的構(gòu)造函數(shù),所以自動(dòng)調(diào)用父類的default構(gòu)造函數(shù))調(diào)用父類的構(gòu)造函數(shù),試圖產(chǎn)生一個(gè)Cleanser class實(shí)例。
11) 產(chǎn)生一個(gè)Base class實(shí)例。先初始化成員變量,再調(diào)用構(gòu)造函數(shù)。
12) 回到Cleanser class,產(chǎn)生一個(gè)實(shí)例。首先初始化Cleanser class中的成員數(shù)據(jù),再執(zhí)行構(gòu)造函數(shù)Cleanser(String a)中的其余部分。