Java向上轉(zhuǎn)型和向下轉(zhuǎn)型實例解析
這篇文章主要介紹了Java向上轉(zhuǎn)型和向下轉(zhuǎn)型實例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
學(xué)習(xí)向上轉(zhuǎn)型和向下轉(zhuǎn)型怎么用沒多難,但是為什么那樣用,我搞了很多次沒弄明白。沒弄明白的原因是平時學(xué)習(xí)時之看例子,而例子一般都比較簡單,沒有對象之間的調(diào)用,一般就是一個對象調(diào)用自己的方法。
首先看下怎么用轉(zhuǎn)型。
要轉(zhuǎn)型,首先要有繼承。繼承是面向?qū)ο笳Z言中一個代碼復(fù)用的機(jī)制,簡單說就是子類繼承了父類中的非私有屬性和可以繼承的方法,然后子類可以繼續(xù)擴(kuò)展自己的屬性及方法。
向上轉(zhuǎn)型:子類對象轉(zhuǎn)為父類,父類可以是接口。公式:Father f = new Son();Father是父類或接口,son是子類。
向下轉(zhuǎn)型:父類對象轉(zhuǎn)為子類。公式:Son s = (Son)f;
例子:向上轉(zhuǎn)型
package multistate;
public class Human {
public void sleep() {
System.out.println("Human sleep..");
}
public static void main(String[] args) {
Human h = new Male();// 向上轉(zhuǎn)型
h.sleep();
Male m = new Male();// 干嘛要向上轉(zhuǎn)型
m.sleep();
// h.speak();此方法不能編譯,報錯說Human類沒有此方法
}
}
class Male extends Human {
@Override
public void sleep() {
System.out.println("Male sleep..");
}
public void speak() {
System.out.println("I am Male");
}
}
class Female extends Human {
@Override
public void sleep() {
System.out.println("Female sleep..");
}
public void speak() {
System.out.println("I am Female");
}
}
注意:向上轉(zhuǎn)型不要強(qiáng)制轉(zhuǎn)型。向上轉(zhuǎn)型后父類的引用所指向的屬性是父類的屬性,如果子類重寫了父類的方法,那么父類引用指向的或者調(diào)用的方法是子類的方法,這個叫動態(tài)綁定。向上轉(zhuǎn)型后父類引用不能調(diào)用子類自己的方法,就是父類沒有但是子類的方法,如果調(diào)用不能編譯通過,比如子類的speak方法。
非要調(diào)用子類的屬性呢?如果不向下轉(zhuǎn)型就需要給需要的屬性寫getter方法。
例子:
package multistate;
public class Human {
String name = "Human";
public String getName(){
return this.name;
}
public void sleep() {
System.out.println("Human sleep..");
}
public static void main(String[] args) {
Human h = new Male();// 向上轉(zhuǎn)型
System.out.println(h.getName());
}
}
class Male extends Human {
String name = "Male";
public String getName(){
return this.name;
}
@Override
public void sleep() {
System.out.println("Male sleep..");
}
public void speak() {
System.out.println("I am Male");
}
}
class Female extends Human {
String name = "Female";
public String getName(){
return this.name;
}
@Override
public void sleep() {
System.out.println("Female sleep..");
}
public void speak() {
System.out.println("I am Female");
}
}
非要調(diào)用子類擴(kuò)展的方法,比如speak方法,就只能向下轉(zhuǎn)型了。
例子:向下轉(zhuǎn)型
向下轉(zhuǎn)型需要考慮安全性,如果父類引用的對象是父類本身,那么在向下轉(zhuǎn)型的過程中是不安全的,編譯不會出錯,但是運行時會出現(xiàn)java.lang.ClassCastException錯誤。它可以使用instanceof來避免出錯此類錯誤即能否向下轉(zhuǎn)型,只有先經(jīng)過向上轉(zhuǎn)型的對象才能繼續(xù)向下轉(zhuǎn)型。
package multistate;
public class Human {
public void sleep() {
System.out.println("Human sleep..");
}
public static void main(String[] args) {
Human h = new Male();// 向上轉(zhuǎn)型
Human h1 = new Human();
//h.speak();此時需要向下轉(zhuǎn)型,否則不能調(diào)用speak方法。
Male m = (Male) h;
m.speak();
/**Male m1 = (Male)h1;
m1.speak(); 此時會出現(xiàn)運行時錯誤,所以可以用instanceOF判斷*/
if (h1 instanceof Male){
Male m1 = (Male)h1;
m1.speak();
}
}
}
class Male extends Human {
@Override
public void sleep() {
System.out.println("Male sleep..");
}
public void speak() {
System.out.println("I am Male");
}
}
弄了半天,向上轉(zhuǎn)型反而不能擁有子類的全部方法,還得向下轉(zhuǎn)型,那直接Son s = new Son();豈不是很方便?不知道是不是就我一個開始學(xué)習(xí)轉(zhuǎn)型有這種想法。
最后搞明白了,原因還是我我的例子太簡單,沒有 考慮過要把類的對象傳遞給其他函數(shù)的例子。
例子:體現(xiàn)向上轉(zhuǎn)型的好處,節(jié)省代碼。
package multistate;
public class Human {
public void sleep() {
System.out.println("Human sleep..");
}
public static void doSleep(Human h){
h.sleep();
}//此時傳遞的參數(shù)是父類對象,但是實際調(diào)用時傳遞子類對象,就是向上轉(zhuǎn)型。
public static void main(String[] args) {
Human h = new Male();// 向上轉(zhuǎn)型
doSleep(new Male());//此處匿名子類對象,當(dāng)然實際應(yīng)用時應(yīng)該是用上面的向上轉(zhuǎn)型公式,然后將子類對象傳遞進(jìn)來,這樣以后好在向下轉(zhuǎn)型,此處沒有向下轉(zhuǎn)型,所以直接用了匿名類對象。
doSleep(new Female());
}
}
class Male extends Human {
@Override
public void sleep() {
System.out.println("Male sleep..");
}
}
class Female extends Human {
@Override
public void sleep() {
System.out.println("Female sleep..");
}
}
如果不向上轉(zhuǎn)型則必須寫兩個doSleep函數(shù),一個傳遞Male類對象,一個傳遞Female類對象。這還是兩個子類,如果有很多子類呢,就要寫很多相同的函數(shù),造成重復(fù)。
好,終于也理解了為什么要向上轉(zhuǎn)型,一旦向上轉(zhuǎn)型了,當(dāng)需要用到子類的方法時,就需要向下轉(zhuǎn)型,也就是為什么要向下轉(zhuǎn)型也解決了。
總結(jié)一下:
1、把子類對象直接賦給父類引用叫upcasting向上轉(zhuǎn)型,向上轉(zhuǎn)型不用強(qiáng)制轉(zhuǎn)型。
如Father father = new Son();
2、把指向子類對象的父類引用賦給子類引用叫向下轉(zhuǎn)型(downcasting),要強(qiáng)制轉(zhuǎn)型,要向下轉(zhuǎn)型,必須先向上轉(zhuǎn)型為了安全可以用instanceof判斷。
如father就是一個指向子類對象的父類引用,把father賦給子類引用son 即Son son =(Son)father;
其中father前面的(Son)必須添加,進(jìn)行強(qiáng)制轉(zhuǎn)換。
3、upcasting 會丟失子類特有的方法,但是子類overriding 父類的方法,子類方法有效,向上轉(zhuǎn)型只能引用父類對象的屬性,要引用子類對象屬性,則要寫getter函數(shù)。
4、向上轉(zhuǎn)型的作用,減少重復(fù)代碼,父類為參數(shù),調(diào)有時用子類作為參數(shù),就是利用了向上轉(zhuǎn)型。這樣使代碼變得簡潔。體現(xiàn)了JAVA的抽象編程思想。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Spring Boot2配置Swagger2生成API接口文檔詳情
這篇文章主要介紹了Spring Boot2配置Swagger2生成API接口文檔詳情,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09
SpringMVC 通過commons-fileupload實現(xiàn)文件上傳功能
這篇文章主要介紹了SpringMVC 通過commons-fileupload實現(xiàn)文件上傳,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-02-02
使用@Valid+BindingResult進(jìn)行controller參數(shù)校驗方式
這篇文章主要介紹了使用@Valid+BindingResult進(jìn)行controller參數(shù)校驗方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12
Java BufferedWriter BufferedReader 源碼分析
本文是關(guān)于Java BufferedWriter ,BufferedReader 簡介、分析源碼 對Java IO 流深入了解,希望看到的同學(xué)對你有所幫助2016-07-07

