重寫(xiě)是子類(lèi)對(duì)父類(lèi)的允許訪問(wèn)的方法的實(shí)現(xiàn)過(guò)程進(jìn)行重新編寫(xiě)!返回值和形參都不能改變。即外殼不變,核心重寫(xiě)!
重寫(xiě)的好處在于子類(lèi)可以根據(jù)需要,定義特定于自己的行為。
也就是說(shuō)子類(lèi)能夠根據(jù)需要實(shí)現(xiàn)父類(lèi)的方法。
在面向?qū)ο笤瓌t里,重寫(xiě)意味著可以重寫(xiě)任何現(xiàn)有方法。實(shí)例如下:
class Animal{
public void move(){
System.out.println("動(dòng)物可以移動(dòng)");
}
}
class Dog extends Animal{
public void move(){
System.out.println("狗可以跑和走");
}
}
public class TestDog{
public static void main(String args[]){
Animal a = new Animal(); // Animal 對(duì)象
Animal b = new Dog(); // Dog 對(duì)象
a.move();// 執(zhí)行 Animal 類(lèi)的方法
b.move();//執(zhí)行 Dog 類(lèi)的方法
}
}
以上實(shí)例編譯運(yùn)行結(jié)果如下:
動(dòng)物可以移動(dòng)
狗可以跑和走
在上面的例子中可以看到,盡管 b 屬于 Animal 類(lèi)型,但是它運(yùn)行的是 Dog 類(lèi)的 move 方法。
這是由于在編譯階段,只是檢查參數(shù)的引用類(lèi)型。
然而在運(yùn)行時(shí),Java 虛擬機(jī) (JVM) 指定對(duì)象的類(lèi)型并且運(yùn)行該對(duì)象的方法。
因此在上面的例子中,之所以能編譯成功,是因?yàn)?Animal 類(lèi)中存在 move 方法,然而運(yùn)行時(shí),運(yùn)行的是特定對(duì)象的方法。
思考以下例子:
class Animal{
public void move(){
System.out.println("動(dòng)物可以移動(dòng)");
}
}
class Dog extends Animal{
public void move(){
System.out.println("狗可以跑和走");
}
public void bark(){
System.out.println("狗可以吠叫");
}
}
public class TestDog{
public static void main(String args[]){
Animal a = new Animal(); // Animal 對(duì)象
Animal b = new Dog(); // Dog 對(duì)象
a.move();// 執(zhí)行 Animal 類(lèi)的方法
b.move();//執(zhí)行 Dog 類(lèi)的方法
a.bark();//執(zhí)行 Animal 類(lèi)的方法
}
}
以上實(shí)例編譯運(yùn)行結(jié)果如下:
TestDog.java:30: cannot find symbol
symbol : method bark()
location: class Animal
a.bark();
^
該程序?qū)伋鲆粋€(gè)編譯錯(cuò)誤,因?yàn)?a 的引用類(lèi)型 Animal 沒(méi)有 bark 方法。
當(dāng)需要在子類(lèi)中調(diào)用父類(lèi)的被重寫(xiě)方法時(shí),要使用 super 關(guān)鍵字。
class Animal{
public void move(){
System.out.println("動(dòng)物可以移動(dòng)");
}
}
class Dog extends Animal{
public void move(){
super.move(); // 應(yīng)用super類(lèi)的方法
System.out.println("狗可以跑和走");
}
}
public class TestDog{
public static void main(String args[]){
Animal b = new Dog(); //
b.move(); //執(zhí)行 Dog類(lèi)的方法
}
}
以上實(shí)例編譯運(yùn)行結(jié)果如下:
動(dòng)物可以移動(dòng)
狗可以跑和走
重載 (overloading) 是在一個(gè)類(lèi)里面,方法名字相同,而參數(shù)不同。返回類(lèi)型呢?可以相同也可以不同。
每個(gè)重載的方法(或者構(gòu)造函數(shù))都必須有一個(gè)獨(dú)一無(wú)二的參數(shù)類(lèi)型列表。
最常用的地方就是構(gòu)造器的重載。
重載規(guī)則
public class Overloading {
public int test(){
System.out.println("test1");
return 1;
}
public void test(int a){
System.out.println("test2");
}
//以下兩個(gè)參數(shù)類(lèi)型順序不同
public String test(int a,String s){
System.out.println("test3");
return "returntest3";
}
public String test(String s,int a){
System.out.println("test4");
return "returntest4";
}
public static void main(String[] args){
Overloading o = new Overloading();
System.out.println(o.test());
o.test(1);
System.out.println(o.test(1,"test3"));
System.out.println(o.test("test4",1));
}
}
以上實(shí)例編譯運(yùn)行結(jié)果如下:
test1
1
test2
test3
returntest3
test4
returntest4
區(qū)別點(diǎn) | 重載方法 | 重寫(xiě)方法 |
---|---|---|
參數(shù)列表 | 必須修改 | 一定不能修改 |
返回類(lèi)型 | 可以修改 | 一定不能修改 |
異常 | 可以修改 | 可以減少或刪除,一定不能拋出新的或者更廣的異常 |
訪問(wèn) | 可以修改 | 一定不能做更嚴(yán)格的限制(可以降低限制) |
方法的重寫(xiě) (Overriding) 和重載 (Overloading) 是 java 多態(tài)性的不同表現(xiàn),重寫(xiě)是父類(lèi)與子類(lèi)之間多態(tài)性的一種表現(xiàn),重載可以理解成多態(tài)的具體表現(xiàn)形式。
更多建議: