重寫(Override)
重寫是子類對(duì)父類的允許訪問的方法的實(shí)現(xiàn)過程進(jìn)行重新編寫!返回值和形參都不能改變。即外殼不變,核心重寫!
重寫的好處在于子類可以根據(jù)需要,定義特定于自己的行為。
也就是說子類能夠根據(jù)需要實(shí)現(xiàn)父類的方法。
在面向?qū)ο笤瓌t里,重寫意味著可以重寫任何現(xiàn)有方法。實(shí)例如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
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 類的方法 b.move(); //執(zhí)行 Dog 類的方法 } } |
以上實(shí)例編譯運(yùn)行結(jié)果如下:
1
2
|
動(dòng)物可以移動(dòng) 狗可以跑和走 |
在上面的例子中可以看到,盡管b屬于Animal類型,但是它運(yùn)行的是Dog類的move方法。
這是由于在編譯階段,只是檢查參數(shù)的引用類型。
然而在運(yùn)行時(shí),Java虛擬機(jī)(JVM)指定對(duì)象的類型并且運(yùn)行該對(duì)象的方法。
因此在上面的例子中,之所以能編譯成功,是因?yàn)锳nimal類中存在move方法,然而運(yùn)行時(shí),運(yùn)行的是特定對(duì)象的方法。
思考以下例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
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 類的方法 b.move(); //執(zhí)行 Dog 類的方法 b.bark(); } } |
以上實(shí)例編譯運(yùn)行結(jié)果如下:
1
2
3
4
|
TestDog.java: 30 : cannot find symbol symbol : method bark() location: class Animal b.bark(); ^ |
該程序?qū)伋鲆粋€(gè)編譯錯(cuò)誤,因?yàn)閎的引用類型Animal沒有bark方法。
方法的重寫規(guī)則
參數(shù)列表必須完全與被重寫方法的相同;
返回類型必須完全與被重寫方法的返回類型相同;
訪問權(quán)限不能比父類中被重寫的方法的訪問權(quán)限更高。例如:如果父類的一個(gè)方法被聲明為public,那么在子類中重寫該方法就不能聲明為protected。
父類的成員方法只能被它的子類重寫。
聲明為final的方法不能被重寫。
聲明為static的方法不能被重寫,但是能夠被再次聲明。
子類和父類在同一個(gè)包中,那么子類可以重寫父類所有方法,除了聲明為private和final的方法。
子類和父類不在同一個(gè)包中,那么子類只能夠重寫父類的聲明為public和protected的非final方法。
重寫的方法能夠拋出任何非強(qiáng)制異常,無論被重寫的方法是否拋出異常。但是,重寫的方法不能拋出新的強(qiáng)制性異常,或者比被重寫方法聲明的更廣泛的強(qiáng)制性異常,反之則可以。
構(gòu)造方法不能被重寫。
如果不能繼承一個(gè)方法,則不能重寫這個(gè)方法。
Super關(guān)鍵字的使用
當(dāng)需要在子類中調(diào)用父類的被重寫方法時(shí),要使用super關(guān)鍵字。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
class Animal{ public void move(){ System.out.println( "動(dòng)物可以移動(dòng)" ); } } class Dog extends Animal{ public void move(){ super .move(); // 應(yīng)用super類的方法 System.out.println( "狗可以跑和走" ); } } public class TestDog{ public static void main(String args[]){ Animal b = new Dog(); // Dog 對(duì)象 b.move(); //執(zhí)行 Dog類的方法 } } |
以上實(shí)例編譯運(yùn)行結(jié)果如下:
1
2
|
動(dòng)物可以移動(dòng) 狗可以跑和走 |
重載(Overload)
重載(overloading) 是在一個(gè)類里面,方法名字相同,而參數(shù)不同。返回類型呢?可以相同也可以不同。
每個(gè)重載的方法(或者構(gòu)造函數(shù))都必須有一個(gè)獨(dú)一無二的參數(shù)類型列表。
只能重載構(gòu)造函數(shù)
重載規(guī)則
被重載的方法必須改變參數(shù)列表;
被重載的方法可以改變返回類型;
被重載的方法可以改變?cè)L問修飾符;
被重載的方法可以聲明新的或更廣的檢查異常;
方法能夠在同一個(gè)類中或者在一個(gè)子類中被重載。
實(shí)例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
public class Overloading { public int test(){ System.out.println( "test1" ); return 1 ; } public void test( int a){ System.out.println( "test2" ); } //以下兩個(gè)參數(shù)類型順序不同 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 )); } } |
重寫與重載之間的區(qū)別
區(qū)別點(diǎn) | 重載方法 | 重寫方法 |
參數(shù)列表 | 必須修改 | 一定不能修改 |
返回類型 | 可以修改 | 一定不能修改 |
異常 | 可以修改 | 可以減少或刪除,一定不能拋出新的或者更廣的異常 |
訪問 | 可以修改 | 一定不能做更嚴(yán)格的限制(可以降低限制) |
希望本篇文章對(duì)各位朋友有所幫助
原文鏈接:http://www.2cto.com/kf/201601/486515.html