實(shí)例分析java中重載與重寫的區(qū)別
本文以實(shí)例詳細(xì)分析了Java中重載與重寫的區(qū)別,感興趣的朋友可以參考一下。
一、重載(Overloading):
(1) 方法重載是讓類以統(tǒng)一的方式處理不同類型數(shù)據(jù)的一種手段。多個同名函數(shù)同時存在,具有不同的參數(shù)個數(shù)/類型。
重載Overloading是一個類中多態(tài)性的一種表現(xiàn)。
(2)Java的方法重載,就是在類中可以創(chuàng)建多個方法,它們具有相同的名字,但具有不同的參數(shù)和不同的定義。
調(diào)用方法時通過傳遞給它們的不同參數(shù)個數(shù)和參數(shù)類型來決定具體使用哪個方法, 這就是多態(tài)性。
(3) 重載的時候,方法名要一樣,但是參數(shù)類型和個數(shù)不一樣,返回值類型可以相同也可以不相同。無法以返回型別作為重載函數(shù)的區(qū)分標(biāo)準(zhǔn)。
下面是重載的例子:
package c04.answer;//這是包名 //這是這個程序的第一種編程方法,在main方法中先創(chuàng)建一個Dog類實(shí)例,然后在Dog類的構(gòu)造方法中利用this關(guān)鍵字調(diào)用不同的bark方法。 不同的重載方法bark是根據(jù)其參數(shù)類型的不同而區(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ù)類型”和“類名”來區(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\");
二、重寫(Overriding)
(1) 父類與子類之間的多態(tài)性,對父類的函數(shù)進(jìn)行重新定義。如果在子類中定義某方法與其父類有相同的名稱和參數(shù),我們說該方法被重寫 (Overriding)。在中,子類可繼承父類中的方法,而不需要重新編寫相同的方法。
但有時子類并不想原封不動地繼承父類的方法,而是想作一定的修改,這就需要采用方法的重寫。
方法重寫又稱方法覆蓋。
(2)若子類中的方法與父類中的某一方法具有相同的方法名、返回類型和參數(shù)表,則新方法將覆蓋原有的方法。
如需父類中原有的方法,可使用super關(guān)鍵字,該關(guān)鍵字引用了當(dāng)前類的父類。
(3)子類函數(shù)的訪問修飾權(quán)限不能少于父類的;
下面是重寫的例子:
概念:即調(diào)用對象方法的機(jī)制。
動態(tài)綁定的內(nèi)幕:
1、編譯器檢查對象聲明的類型和方法名,從而獲取所有候選方法。試著把上例Base類的test注釋掉,這時再編譯就無法通過。
2、重載決策:編譯器檢查方法調(diào)用的參數(shù)類型,從上述候選方法選出唯一的那一個(其間會有隱含類型轉(zhuǎn)化)。
如果編譯器找到多于一個或者沒找到,此時編譯器就會報錯。試著把上例Base類的test(byte b)注釋掉,這時運(yùn)行結(jié)果是1 1。
3、若方法類型為priavte static final ,java采用靜態(tài)編譯,編譯器會準(zhǔn)確知道該調(diào)用哪
個方法。
4、當(dāng)程序運(yùn)行并且使用動態(tài)綁定來調(diào)用一個方法時,那么虛擬機(jī)必須調(diào)用對象的實(shí)際類型相匹配的方法版本。
在例子中,b所指向的實(shí)際類型是TestOverriding,所以b.test(0)調(diào)用子類的test。
但是,子類并沒有重寫test(byte b),所以b.test((byte)0)調(diào)用的是父類的test(byte b)。
如果把父類的(byte b)注釋掉,則通過第二步隱含類型轉(zhuǎn)化為int,最終調(diào)用的是子類的test(int i)。
三、學(xué)習(xí)總結(jié):
多態(tài)性是面向?qū)ο缶幊痰囊环N特性,和方法無關(guān),
簡單說,就是同樣的一個方法能夠根據(jù)輸入數(shù)據(jù)的不同,做出不同的處理,即方法的
重載——有不同的參數(shù)列表(靜態(tài)多態(tài)性)
而當(dāng)子類繼承自父類的相同方法,輸入數(shù)據(jù)一樣,但要做出有別于父類的響應(yīng)時,你就要覆蓋父類方法,
即在子類中重寫該方法——相同參數(shù),不同實(shí)現(xiàn)(動態(tài)多態(tài)性)
OOP三大特性:繼承,多態(tài),封裝。
public class Base { void test(int i) { System.out.print(i); } void test(byte b) { System.out.print(b); } } public class TestOverriding extends Base { void test(int i) { i++; System.out.println(i); } public static void main(String[]agrs) { Base b=new TestOverriding(); b.test(0) b.test((byte)0) } }
這時的輸出結(jié)果是1 0,這是運(yùn)行時動態(tài)綁定的結(jié)果。
重寫的主要優(yōu)點(diǎn)是能夠定義某個子類特有的特征:
publicclassFather{ publicvoidspeak(){ System.out.println(Father); } } publicclassSonextendsFather{ publicvoidspeak(){ System.out.println("son"); } }
這也叫做多態(tài)性,重寫方法只能存在于具有繼承關(guān)系中,重寫方法只能重寫父類非私有的方法。
當(dāng)上例中Father類speak()方法被private時,Son類不能重寫出Father類speak()方法,此時Son類speak()方法相當(dāng)與在Son類中定義的一個speak()方法。
Father類speak()方法一但被final時,無論該方法被public,protected及默認(rèn)所修飾時,Son類根本不能重寫Father類speak()方法,
試圖編譯代碼時,編譯器會報錯。例:
publicclassFather{ finalpublicvoidspeak(){ System.out.println("Father"); } } publicclassSonextendsFather{ publicvoidspeak(){ System.out.println("son"); } }//編譯器會報錯;
Father類speak()方法被默認(rèn)修飾時,只能在同一包中,被其子類被重寫,如果不在同一包則不能重寫。
Father類speak()方法被protoeted時,不僅在同一包中,被其子類被重寫,還可以不同包的子類重寫。
重寫方法的規(guī)則:
1、參數(shù)列表必須完全與被重寫的方法相同,否則不能稱其為重寫而是重載。
2、返回的類型必須一直與被重寫的方法的返回類型相同,否則不能稱其為重寫而是重載。
3、訪問修飾符的限制一定要大于被重寫方法的訪問修飾符(public>protected>default>private)
4、重寫方法一定不能拋出新的檢查異常或者比被重寫方法申明更加寬泛的檢查型異常。例如:
父類的一個方法申明了一個檢查異常IOException,在重寫這個方法是就不能拋出Exception,只能拋出IOException的子類異常,可以拋出非檢查異常。
而重載的規(guī)則:
1、必須具有不同的參數(shù)列表;
2、可以有不責(zé)罵的返回類型,只要參數(shù)列表不同就可以了;
3、可以有不同的訪問修飾符;
4、可以拋出不同的異常;
重寫與重載的區(qū)別在于:
重寫多態(tài)性起作用,對調(diào)用被重載過的方法可以大大減少代碼的輸入量,同一個方法名只要往里面?zhèn)鬟f不同的參數(shù)就可以擁有不同的功能或返回值。
用好重寫和重載可以設(shè)計一個結(jié)構(gòu)清晰而簡潔的類,可以說重寫和重載在編寫代碼過程中的作用非同一般.
相關(guān)文章
Java8中利用stream對map集合進(jìn)行過濾的方法
這篇文章主要給大家介紹了關(guān)于Java8中利用stream對map集合進(jìn)行過濾的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-07-07MyBatis中的collection兩種使用方法及效率比較
collection主要是應(yīng)對表關(guān)系是一對多的情況,本文主要介紹了MyBatis中的collection兩種使用方法及效率比較,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06淺談Java 三種方式實(shí)現(xiàn)接口校驗(yàn)
這篇文章主要介紹了淺談Java 三種方式實(shí)現(xiàn)接口校驗(yàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-10-10深入理解java異常處理機(jī)制的原理和開發(fā)應(yīng)用
Java異常處理機(jī)制在日常開發(fā)中應(yīng)用頻繁,本篇文章主要在基礎(chǔ)的使用方法上,更進(jìn)一步的,如何更加合理的使用異常機(jī)制,希望可以對各位朋友能有所幫助。2017-04-04spring Boot 應(yīng)用通過Docker 來實(shí)現(xiàn)構(gòu)建、運(yùn)行、發(fā)布流程
這篇文章主要介紹了spring Boot 應(yīng)用通過Docker 來實(shí)現(xiàn)構(gòu)建、運(yùn)行、發(fā)布流程,圖文詳解,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-11-11@Scheduled定時器原理及@RefreshScope相互影響
這篇文章主要為大家介紹了@Scheduled定時器原理及@RefreshScope相互影響詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07詳解MyBatis?ResultSetHandler?結(jié)果集的解析過程
這篇文章主要為大家介紹了MyBatis?ResultSetHandler?結(jié)果集的解析過程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02