漫談java中重載與重寫(xiě)的不同
推薦 + 挑錯(cuò) + 收藏(0) + 用戶(hù)評(píng)論(0)
首先我們來(lái)講講:重載(Overloading)
?。?) 方法重載是讓類(lèi)以統(tǒng)一的方式處理不同類(lèi)型數(shù)據(jù)的一種手段。多個(gè)同名函數(shù)同時(shí)存在,具有不同的參數(shù)個(gè)數(shù)/類(lèi)型。
重載Overloading是一個(gè)類(lèi)中多態(tài)性的一種表現(xiàn)。
?。?) Java的方法重載,就是在類(lèi)中可以創(chuàng)建多個(gè)方法,它們具有相同的名字,但具有不同的參數(shù)和不同的定義。
調(diào)用方法時(shí)通過(guò)傳遞給它們的不同參數(shù)個(gè)數(shù)和參數(shù)類(lèi)型來(lái)決定具體使用哪個(gè)方法, 這就是多態(tài)性。
(3) 重載的時(shí)候,方法名要一樣,但是參數(shù)類(lèi)型和個(gè)數(shù)不一樣,返回值類(lèi)型可以相同也可以不相同。無(wú)法以返回型別作為重載函數(shù)的區(qū)分標(biāo)準(zhǔn)。
下面是重載的例子:
package c04.answer;//這是包名 //這是這個(gè)程序的第一種編程方法,在main方法中先創(chuàng)建一個(gè)Dog類(lèi)實(shí)例,然后在Dog類(lèi)的構(gòu)造方法中利用this關(guān)鍵字調(diào)用不同的bark方法。 不同的重載方法bark是根據(jù)其參數(shù)類(lèi)型的不同而區(qū)分的。 //注意:除構(gòu)造器以外,編譯器禁止在其他任何地方中調(diào)用構(gòu)造器。 package c04.answer; public class Dog {Dog() {this.bark();}void bark()//bark()方法是重載方法 {System.out.println( \“no barking! \”); this.bark( \“female\”, 3.4); }void bark(String m,double l)//注意:重載的方法的返回值都是一樣的,{System.out.println( \“a barking dog! \”); this.bark(5, \“China \”); }void bark(int a,String n)//不能以返回值區(qū)分重載方法,而只能以“參數(shù)類(lèi)型”和“類(lèi)名”來(lái)區(qū)分 {System.out.println( \“a howling dog \”); }public static void main(String []args) {Dog dog = new Dog(); //dog.bark();[Page ]//dog.bark( \“male \”, \“yellow \”); //dog.bark(5, \“China \”);
然后我們?cè)賮?lái)談?wù)?重寫(xiě)(Overriding)
?。?) 父類(lèi)與子類(lèi)之間的多態(tài)性,對(duì)父類(lèi)的函數(shù)進(jìn)行重新定義。如果在子類(lèi)中定義某方法與其父類(lèi)有相同的名稱(chēng)和參數(shù),我們說(shuō)該方法被重寫(xiě) (Overriding)。在Java中,子類(lèi)可繼承父類(lèi)中的方法,而不需要重新編寫(xiě)相同的方法。
但有時(shí)子類(lèi)并不想原封不動(dòng)地繼承父類(lèi)的方法,而是想作一定的修改,這就需要采用方法的重寫(xiě)。
方法重寫(xiě)又稱(chēng)方法覆蓋。
(2)若子類(lèi)中的方法與父類(lèi)中的某一方法具有相同的方法名、返回類(lèi)型和參數(shù)表,則新方法將覆蓋原有的方法。
如需父類(lèi)中原有的方法,可使用super關(guān)鍵字,該關(guān)鍵字引用了當(dāng)前類(lèi)的父類(lèi)。
(3)子類(lèi)函數(shù)的訪問(wèn)修飾權(quán)限不能少于父類(lèi)的; 下面是重寫(xiě)的例子:
概念:即調(diào)用對(duì)象方法的機(jī)制。
動(dòng)態(tài)綁定的內(nèi)幕:
1、編譯器檢查對(duì)象聲明的類(lèi)型和方法名,從而獲取所有候選方法。試著把上例Base類(lèi)的test注釋掉,這時(shí)再編譯就無(wú)法通過(guò)。
2、重載決策:編譯器檢查方法調(diào)用的參數(shù)類(lèi)型,從上述候選方法選出唯一的那一個(gè)(其間會(huì)有隱含類(lèi)型轉(zhuǎn)化)。
如果編譯器找到多于一個(gè)或者沒(méi)找到,此時(shí)編譯器就會(huì)報(bào)錯(cuò)。試著把上例Base類(lèi)的test(byte b)注釋掉,這時(shí)運(yùn)行結(jié)果是1 1。
3、若方法類(lèi)型為priavte static final ,java采用靜態(tài)編譯,編譯器會(huì)準(zhǔn)確知道該調(diào)用哪 個(gè)方法。
4、當(dāng)程序運(yùn)行并且使用動(dòng)態(tài)綁定來(lái)調(diào)用一個(gè)方法時(shí),那么虛擬機(jī)必須調(diào)用對(duì)象的實(shí)際類(lèi)型相匹配的方法版本。
在例子中,b所指向的實(shí)際類(lèi)型是TestOverriding,所以b.test(0)調(diào)用子類(lèi)的test。
但是,子類(lèi)并沒(méi)有重寫(xiě)test(byte b),所以b.test((byte)0)調(diào)用的是父類(lèi)的test(byte b)。
如果把父類(lèi)的(byte b)注釋掉,則通過(guò)第二步隱含類(lèi)型轉(zhuǎn)化為int,最終調(diào)用的是子類(lèi)的test(int i)。
學(xué)習(xí)總結(jié):
多態(tài)性是面向?qū)ο缶幊痰囊环N特性,和方法無(wú)關(guān), 簡(jiǎn)單說(shuō),就是同樣的一個(gè)方法能夠根據(jù)輸入數(shù)據(jù)的不同,做出不同的處理,即方法的 重載——有不同的參數(shù)列表(靜態(tài)多態(tài)性)
而當(dāng)子類(lèi)繼承自父類(lèi)的相同方法,輸入數(shù)據(jù)一樣,但要做出有別于父類(lèi)的響應(yīng)時(shí),你就要覆蓋父類(lèi)方法,
即在子類(lèi)中重寫(xiě)該方法——相同參數(shù),不同實(shí)現(xiàn)(動(dòng)態(tài)多態(tài)性)
OOP三大特性:繼承,多態(tài),封裝。
/** * Java學(xué)習(xí)交流QQ群:589809992 我們一起學(xué)Java! */publicclassBase{voidtest( inti) { System.out.print(i); } voidtest( byteb) { System.out.print(b); } }publicclassTestOverridingextendsBase{voidtest( inti) { i++; System.out.println(i); }publicstaticvoidmain(String[]agrs) { Base b= newTestOverriding(); b.test( 0) b.test(( byte)0) } }
這時(shí)的輸出結(jié)果是1 0,這是運(yùn)行時(shí)動(dòng)態(tài)綁定的結(jié)果。
重寫(xiě)的主要優(yōu)點(diǎn)是能夠定義某個(gè)子類(lèi)特有的特征:
publicclassFather{publicvoidspeak(){ System.out.println(Father); } }publicclassSonextendsFather{publicvoidspeak(){ System.out.println( “son”); } }
這也叫做多態(tài)性,重寫(xiě)方法只能存在于具有繼承關(guān)系中,重寫(xiě)方法只能重寫(xiě)父類(lèi)非私有的方法。
當(dāng)上例中Father類(lèi)speak()方法被private時(shí),Son類(lèi)不能重寫(xiě)出Father類(lèi)speak()方法,此時(shí)Son類(lèi)speak()方法相當(dāng)與在Son類(lèi)中定義的一個(gè)speak()方法。
Father類(lèi)speak()方法一但被final時(shí),無(wú)論該方法被public,protected及默認(rèn)所修飾時(shí),Son類(lèi)根本不能重寫(xiě)Father類(lèi)speak()方法,
試圖編譯代碼時(shí),編譯器會(huì)報(bào)錯(cuò)。例:
/** * Java學(xué)習(xí)交流QQ群:589809992 我們一起學(xué)Java! */publicclassFather{finalpublicvoidspeak(){ System.out.println( “Father”); } }publicclassSonextendsFather{publicvoidspeak(){ System.out.println( “son”); } } //編譯器會(huì)報(bào)錯(cuò);
Father類(lèi)speak()方法被默認(rèn)修飾時(shí),只能在同一包中,被其子類(lèi)被重寫(xiě),如果不在同一包則不能重寫(xiě)。
Father類(lèi)speak()方法被protoeted時(shí),不僅在同一包中,被其子類(lèi)被重寫(xiě),還可以不同包的子類(lèi)重寫(xiě)。
重寫(xiě)方法的規(guī)則:
1、參數(shù)列表必須完全與被重寫(xiě)的方法相同,否則不能稱(chēng)其為重寫(xiě)而是重載。
2、返回的類(lèi)型必須一直與被重寫(xiě)的方法的返回類(lèi)型相同,否則不能稱(chēng)其為重寫(xiě)而是重載。
3、訪問(wèn)修飾符的限制一定要大于被重寫(xiě)方法的訪問(wèn)修飾符(public》protected》default》private)
4、重寫(xiě)方法一定不能拋出新的檢查異常或者比被重寫(xiě)方法申明更加寬泛的檢查型異常。例如:
父類(lèi)的一個(gè)方法申明了一個(gè)檢查異常IOException,在重寫(xiě)這個(gè)方法是就不能拋出Exception,只能拋出IOException的子類(lèi)異常,可以拋出非檢查異常。
而重載的規(guī)則:
1、必須具有不同的參數(shù)列表;
2、可以有不責(zé)罵的返回類(lèi)型,只要參數(shù)列表不同就可以了;
3、可以有不同的訪問(wèn)修飾符;
4、可以拋出不同的異常;
重寫(xiě)與重載的區(qū)別在于:
重寫(xiě)多態(tài)性起作用,對(duì)調(diào)用被重載過(guò)的方法可以大大減少代碼的輸入量,同一個(gè)方法名只要往里面?zhèn)鬟f不同的參數(shù)就可以擁有不同的功能或返回值。
用好重寫(xiě)和重載可以設(shè)計(jì)一個(gè)結(jié)構(gòu)清晰而簡(jiǎn)潔的類(lèi),可以說(shuō)重寫(xiě)和重載在編寫(xiě)代碼過(guò)程中的作用非同一般
非常好我支持^.^
(0) 0%
不好我反對(duì)
(0) 0%