在說ClassCastException之前,先介紹下引用類型轉換;
引用類型轉換分為向上轉型和向下轉型兩種;
向上轉型:多態本身是子類類型向父類類型向上轉換的過程,這個過程是默認的;當父類引用指向一個子類對象時,便是向上轉換;
使用格式:
父類類型 變量名 = new 子類類型();
向下轉型:父類類型向子類類型向下轉換的過程,這個過程時強制;一個已經向上轉型的子類對象,將父類引用轉為子類引用,可以使用強制轉換的格式,便是向下轉換;
使用格式:
子類類型 變量名 = (子類類型) 父類變量名;
轉型的目的:
當使用多態方式調用方法時,首先檢查父類是否有該方法,如果沒有,則編譯錯誤,即父類不能調用子類擁有的,而父類沒有的方法;編譯都錯誤,更別說運行,要想調用子類特有的方法,必須做向下轉型;
轉型示例代碼:
定義類:
abstract class Animal {
abstract void eat();
}
class Cat extends Animal {
public void eat() {
System.out.println("吃魚");
}
public void catchMouse() {
System.out.println("抓老鼠");
}
}
class Dog extends Animal {
public void eat() {
System.out.println("吃骨頭");
}
public void watchHouse() {
System.out.println("看家");
}
}
測試類:
public class Test {
public static void main(String[] args) {
// 向上轉型
Animal a = new Cat();
// 調用的是 Cat 的 eat
a.eat();
// 向下轉型
Cat c = (Cat)a;
// 調用的是 Cat 的 catchMouse
c.catchMouse();
// 向下轉型
Dog d = (Dog)a;
// ClassCastException異常
d.watchHouse();
}
}
上面這段代碼可以編譯,但在運行時,會報ClassCastException異常;
ClassCastException: 當試圖將對象強制轉換為不是實例的子類時,拋出該異常;
上面的測試類中創建了Cat類型對象,運行時不能轉換成Dog對象的,這兩個類型並沒有任何繼承關系,不符合類型轉換的定義 ,因此會拋出ClassCastException異常;
為了避免ClassCastException的發生,Java提供了 instanceof 關鍵字,給引用變量做類型的校驗,格式如下 :
變量名 instanceof 數據類型
如果變量屬於該數據類型,返回true;如果變量不屬於該數據類型,返回false;
可以使用instanceof將上面的測試類修改如下:
public class Test {
public static void main(String[] args) {
// 向上轉型
Animal a = new Cat();
// 調用的是 Cat 的 eat
a.eat();
// 向下轉型
if (a instanceof Cat) {
Cat c = (Cat)a;
c.catchMouse();
} else if (a instanceof Dog) {
Dog d = (Dog)a;
d.watchHouse();
}
}
}
