Java中你真的會用Constructor構(gòu)造器嗎之看完本篇你就真的會了
引言
相信大家對于java里的構(gòu)造器應(yīng)該都是有了解的,這次我們來了解一些構(gòu)造器的不同使用方式,了解構(gòu)造器的調(diào)用順序,最后可以靈活的在各種情況下定義使用構(gòu)造器,進(jìn)一步優(yōu)化我們的代碼;
構(gòu)造器簡介
還是簡單介紹一下構(gòu)造器到底是什么吧,
構(gòu)造器是類中一種特殊的方法,通過調(diào)用構(gòu)造器來完成對象的創(chuàng)建,以及對象屬性的初始化操作。
構(gòu)造器定義方式:
[修飾符列表] 構(gòu)造器名(形式參數(shù)列表){
構(gòu)造方法體;
}
構(gòu)造器有以下幾個特點:
- 構(gòu)造器名和類名一致;
- 構(gòu)造器用來創(chuàng)建對象,以及完成屬性初始化操作;
- 構(gòu)造器返回值類型不需要寫,包括 void 也不能寫
- 構(gòu)造器的返回值類型實際上是當(dāng)前類的類型
- 一個類中可以定義多個構(gòu)造器,這些構(gòu)造器構(gòu)成方法重載
構(gòu)造器的使用其實很常見,調(diào)用語法為:new 構(gòu)造器名(實際參數(shù)列表) ,假如我們現(xiàn)在有一個Person類,調(diào)用默認(rèn)構(gòu)造器:
Person p = new Person(); // 這里的Person()其實就是調(diào)用的默認(rèn)構(gòu)造器??!
構(gòu)造器的使用需要注意:
當(dāng)一個類沒有顯式的定義任何構(gòu)造器的時候,系統(tǒng)默認(rèn)提供無參數(shù)構(gòu)造器,當(dāng)顯式的定義構(gòu)造器之后,系統(tǒng)不再提供無參數(shù)構(gòu)造器。無參數(shù)構(gòu)造器又叫做缺省構(gòu)造器,或默認(rèn)構(gòu)造器;
構(gòu)造器重載
雖然系統(tǒng)會給我們提供默認(rèn)構(gòu)造器并賦予初始值,我們自己也可以顯式的指定初始值,但是可不可以多種構(gòu)造器同時存在一個類中呢?
這就需要重載構(gòu)造器,因為構(gòu)造器也是方法的一種,方法都可以重載,構(gòu)造器當(dāng)然也可以;
構(gòu)造器重載方式和方法重載差不多,只需要構(gòu)造器名相同,參數(shù)列表不同就可以了;
測試代碼:
public class ConstructorOverloadTest {
String name;
int age;
// 默認(rèn)構(gòu)造器(無參構(gòu)造器)
ConstructorOverloadTest() {}
// 含參構(gòu)造器 重載版
ConstructorOverloadTest(String myName, int myAge) {
this.name = myName;
this.age = myAge;
}
public static void main(String[] args) {
// 調(diào)用無參構(gòu)造器
var p1 = new ConstructorOverloadTest();
System.out.println("p1的初始化結(jié)果為:");
System.out.println("姓名 :" + p1.name);
System.out.println("年齡:" + p1.age);
// 調(diào)用有參構(gòu)造器
var p2 = new ConstructorOverloadTest("張三", 18);
System.out.println("p2的初始化結(jié)果為:");
System.out.println("姓名 :" + p2.name);
System.out.println("年齡:" + p2.age);
}
}
輸出結(jié)果為:
p1的初始化結(jié)果為:
姓名 :null
年齡:0
p2的初始化結(jié)果為:
姓名 :張三
年齡:18
當(dāng)然這里只寫了一個重載版本,還可以定義多個重載版本,這里就不列舉了;我們只需要能夠根據(jù)實際情況使用恰當(dāng)?shù)闹剌d就可以了,不用都寫出來;
構(gòu)造器間的相互調(diào)用
如果系統(tǒng)中有多個構(gòu)造器,并且我們想要通過一個構(gòu)造器A調(diào)用另一個構(gòu)造器B,該怎么辦?
可能有人會想到調(diào)用構(gòu)造器肯定要創(chuàng)建一個對象啊,所以直接new一個不就可以了嘛; 但是我們通過一個構(gòu)造器調(diào)用另一個構(gòu)造器的目的就是為了簡化代碼的,這樣反而又多此一舉了;
如果不new一個新的對象,那么我們可以通過 this引用 來調(diào)用相應(yīng)的構(gòu)造器;
還是不太明白?看看下面的代碼!
測試代碼如下:
public class ConstructorTest01 {
public static void main(String[] args) {
// 調(diào)用兩個參數(shù)的構(gòu)造器
Person p1 = new Person("張三", 18);
System.out.println("兩個參數(shù)的構(gòu)造器初始化結(jié)果為:");
System.out.println("姓名:" + p1.name);
System.out.println("身份證號:" + p1.idNum);
System.out.println("年齡:" + p1.age);
// 調(diào)用三個參數(shù)的構(gòu)造器
Person p2 = new Person("李四", "6666666", 20);
System.out.println("三個參數(shù)的構(gòu)造器初始化結(jié)果為:");
System.out.println("姓名:" + p2.name);
System.out.println("身份證號:" + p2.idNum);
System.out.println("年齡:" + p2.age);
}
}
class Person {
String name;
String idNum;
int age;
// 有兩個參數(shù)的構(gòu)造器
Person(String myName, int myAge) {
this.name = myName;
this.age = myAge;
}
// 有三個參數(shù)的構(gòu)造器
Person(String myName, String myIdNum, int myAge) {
// 通過this調(diào)用兩個參數(shù)的構(gòu)造器
this(myName, myAge);
this.idNum = myIdNum;
}
}
輸出結(jié)果為:
兩個參數(shù)的構(gòu)造器初始化結(jié)果為:
姓名:張三
身份證號:null
年齡:18
三個參數(shù)的構(gòu)造器初始化結(jié)果為:
姓名:李四
身份證號:6666666
年齡:20
通過this調(diào)用有一點需要注意:
使用this調(diào)用另一個重載的構(gòu)造器只能在構(gòu)造器中使用,且必須作為構(gòu)造器執(zhí)行體的第一條語句
可能有人會問:如果在有三個參數(shù)的構(gòu)造器里面不用this指針,而是直接把上一個構(gòu)造器的代碼復(fù)制過來不行嗎?
當(dāng)然可以,但是不建議;我們得知道為什么要存在this這個調(diào)用構(gòu)造器的方法,它不是無緣無故存在的;在軟件開發(fā)時有一個原則:不要把相同代碼重復(fù)寫兩次以上;因為一旦一個構(gòu)造器內(nèi)的初始代碼需要修改,那么很有可能其他的構(gòu)造器初始代碼也要跟著修改,如果構(gòu)造版本多的話是非常麻煩的;而使用this恰恰可以避免這個問題,只需要修改一個地方,其他幾個構(gòu)造器就隨著更改了;這樣的代碼不僅更加簡潔,而且降低了維護(hù)成本;
子類構(gòu)造器調(diào)用父類構(gòu)造器
繼承有一個規(guī)則是:當(dāng)繼承發(fā)生后,子類不會獲得父類的構(gòu)造器;
但是子類的構(gòu)造器可以通過一定方法區(qū)調(diào)用父類構(gòu)造器,類似于通過this引用調(diào)用重載的構(gòu)造器;
而子類構(gòu)造器調(diào)用父類的構(gòu)造器不用this關(guān)鍵字,這里有一個新的關(guān)鍵字:super
super和this用法差不多,使用它就可以實現(xiàn)子類構(gòu)造器調(diào)用父類的構(gòu)造器;
看個代碼就明白了:
測試代碼:
class Person {
String name;
int age;
// 父類的構(gòu)造函數(shù)
Person(String myName, int myAge) {
this.name = myName;
this.age = myAge;
}
}
public class Kids extends Person{
String school;
// 子類的構(gòu)造函數(shù)
Kids(String myName, int myAge, String mySchool) {
// 通過super來調(diào)用父類構(gòu)造器初始化過程
super(myName, myAge);
this.school = mySchool;
}
public static void main(String[] args) {
Kids k = new Kids("張三", 5, "奧特曼小學(xué)");
System.out.println("name:" + k.name);
System.out.println("age:" + k.age);
System.out.println("school:" + k.school);
}
}
輸出結(jié)果為:
name:張三
age:5
school:奧特曼小學(xué)
通過代碼可以看出來super確實和this非常像,無非就是super調(diào)用父類構(gòu)造器,this調(diào)用同一個類的重載的構(gòu)造器;
且使用super也需要注意:
super調(diào)用父類構(gòu)造器也必須在子類構(gòu)造器執(zhí)行體的第一行,所以this和super不會同時出現(xiàn);
構(gòu)造器的調(diào)用順序
這里先看一個代碼:
class Creature {
Creature() {
System.out.println("Creature的無參構(gòu)造器的調(diào)用");
}
}
class Person extends Creature {
Person(String name) {
System.out.println("Person的一個參數(shù)構(gòu)造器的調(diào)用");
}
Person(String name, int age) {
// this調(diào)用同一個類中重載的構(gòu)造器
this(name);
System.out.println("Person的兩個參數(shù)構(gòu)造器的調(diào)用");
}
}
public class Kids02 extends Person {
Kids02() {
// 顯式調(diào)用父類的兩個參數(shù)的構(gòu)造器
super("張三", 18);
System.out.println("Kids02的無參構(gòu)造器的調(diào)用");
}
public static void main(String[] args) {
// 創(chuàng)建一個Kids02對象
new Kids02();
}
}
這個代碼中,Person類繼承了Creature類,Kids02類繼承了Person類,Perosn類中又通過this調(diào)用同一個類中重載的構(gòu)造器,
Kids02類中用super 顯式調(diào)用父類的兩個參數(shù)的構(gòu)造器;
那么輸出是什么呢?到底先調(diào)用的是誰的構(gòu)造器,最后調(diào)用的誰的構(gòu)造器?
輸出如下:
Creature的無參構(gòu)造器的調(diào)用
Person的一個參數(shù)構(gòu)造器的調(diào)用
Person的兩個參數(shù)構(gòu)造器的調(diào)用
Kids02的無參構(gòu)造器的調(diào)用
通過這個代碼和結(jié)果,可以總結(jié)出來,父類構(gòu)造器總在子類構(gòu)造器之前執(zhí)行,或者可以說:
創(chuàng)建任何對象總是從該類所在繼承樹最頂層類的構(gòu)造器開始執(zhí)行的,然后依次向下執(zhí)行;
對于this和super也是一樣;
這個代碼好好理解一下,就會對構(gòu)造器間的關(guān)系體會更深;
總結(jié)
對象的初始化離不開構(gòu)造器,對構(gòu)造器的學(xué)習(xí)真正需要的還是大量的練習(xí),這些內(nèi)容也是需要熟練掌握的;這關(guān)乎著你在實際開發(fā)應(yīng)用過程中對整個代碼的影響和對象能否恰到好處的初始化;希望你能夠真正學(xué)會并應(yīng)用到這些方法,且通過這篇文章能有所收獲!
ps:如果內(nèi)容有問題歡迎討論?。?/p>
到此這篇關(guān)于Java中你真的會用Constructor構(gòu)造器嗎之看完本篇你就真的會了的文章就介紹到這了,更多相關(guān)Java 構(gòu)造器 Constructor內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring事務(wù)&Spring整合MyBatis的兩種方式
這篇文章主要介紹了Spring事務(wù)&Spring整合MyBatis的兩種方式,本文結(jié)合實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-02-02
詳解SpringBoot 應(yīng)用如何提高服務(wù)吞吐量
這篇文章主要介紹了Spring Boot 應(yīng)用如何提高服務(wù)吞吐量,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
hadoop?詳解如何實現(xiàn)數(shù)據(jù)排序
在很多業(yè)務(wù)場景下,需要對原始的數(shù)據(jù)讀取分析后,將輸出的結(jié)果按照指定的業(yè)務(wù)字段進(jìn)行排序輸出,方便上層應(yīng)用對結(jié)果數(shù)據(jù)進(jìn)行展示或使用,減少二次排序的成本2022-02-02
深度解析Spring內(nèi)置作用域及其在實踐中的應(yīng)用
這篇文章主要詳細(xì)介紹了Spring內(nèi)置的作用域類型及其在實踐中的應(yīng)用,文中有詳細(xì)的代碼示例,對我們的餓學(xué)習(xí)或工作有一定的參考價值,感興趣的同學(xué)可以借鑒閱讀2023-06-06

