
我們現在要來詳細討論建構子的作用與功能,建構子的主要目的在於建立物件的過程中,也就是使用 new 關鍵字 (keyword) 的時候,程式 (program) 建立物件所執行的工作,這些工作通常就是初始化物件的屬性 (field) 。
如果類別沒有定義建構子,編譯器會提供預設預購子,預設建構子為沒有參數 (parameter) 的版本
public Encrypt() {}須注意,如果我們有在類別中定義建構子,編譯器便不會提供沒有參數版本的預設建構子。建構子可以依需求而有不同的參數版本,例如 Encrypt 中我們需要這些屬性
char[] cArray; int a; int b; int n; char c; int diff;
其中 a 、 b 、 n 、 c 、 diff ,我們需要提供這五個屬性值,然後呼叫 setArray() 計算出 cArray 的值
public Encrypt(int a, int b, int n, char c, int dif) {
this.a = a;
this.b = b;
this.n = n;
this.c = c;
this.diff = dif;
setArray();
}當然,有些屬性值也可以用預設的,這就是形成不同參數版本的建構子
public Encrypt(int a, int b) {
this(a, b, 26, 'a', 97);
}這是利用 this 呼叫最多參數版本的建構子, a 、 b 為此建構子所需要提供的參數,其餘三個 n 、 c 、 diff ,我們直接以字面常數 (literal) 代入預設值, 也就是 n 為 26 , c 為 'a' , diff 為 97 。
可是 Encrypt 不是要 a 、 b 都為 0 到 9 隨機整數的隨機值嗎?我們仍得在 EncryptDemo 中進行設定嗎?不,不需要,我們可以設計沒有參數版本的建構子
public Encrypt() {
int ta = 0;
int tb = 0;
while (ta % 2 == 0) {
ta = (int) (Math.random() * 10);
tb = (int) (Math.random() * 10);
}
this.a = ta;
this.b = tb;
this.n = 26;
this.c = 'a';
this.diff = 97;
setArray();
}這裡,屬性設定的部份也可以這樣寫
a = ta; b = tb; n = 26; c = 'a'; diff = 97; setArray();
我們並沒有用到與屬性名稱相同的區域變數 (local variable) ,所以可以直接以屬性名稱設定數性質。我們仍使用 this 的原因很簡單,就是既然前兩個建構子都有用到 this ,我們這裡維持一致性,所以也用 this 來設定屬性。
完整的 Encrypt 如下
public class Encrypt {
// 屬性
char[] cArray = new char[26];
int a;
int b;
int n;
char c;
int diff;
// 建構子
public Encrypt(int a, int b, int n, char c, int dif) {
this.a = a;
this.b = b;
this.n = n;
this.c = c;
this.diff = dif;
setArray();
}
public Encrypt(int a, int b) {
this(a, b, 26, 'a', 97);
}
public Encrypt() {
int ta = 0;
int tb = 0;
while (ta % 2 == 0) {
ta = (int) (Math.random() * 10);
tb = (int) (Math.random() * 10);
}
this.a = ta;
this.b = tb;
this.n = 26;
this.c = 'a';
this.diff = 97;
setArray();
}
// setter 方法
public void setArray() {
int i, y, x, m;
char r;
x = c;
for (i = 0; i < n; i++) {
y = x * a + b;
m = y % n;
r = (char) (m + diff);
cArray[i] = r;
x++;
}
}
}
/* 《程式語言教學誌》的範例程式
http://pydoing.blogspot.com/
檔名:Encrypt.java
功能:示範 Java 程式
作者:張凱慶
時間:西元 2011 年 4 月 */然後修改 EncryptDemo 進行測試
// 測試 Encrypt 的類別
public class EncryptDemo {
public static void main(String[] args) {
System.out.println();
for (int i = 0; i < 16; i++) {
Encrypt t = new Encrypt();
System.out.println(t.cArray);
}
System.out.println();
}
}
/* 《程式語言教學誌》的範例程式
http://pydoing.blogspot.com/
檔名:EncryptDemo.java
功能:示範 Java 程式
作者:張凱慶
時間:西元 2011 年 4 月 */記得 Encrypt 與 EncryptDemo 都修改過,因此執行前都要先進行編譯,如下

這裡,我們在 EncryptDemo 中直接用句點運算子存取屬性 cArray ,其實不一個好習慣,因為如此物件的資料就曝光在世人面前,任何人都可以透過句點運算子存取,甚至更改屬性值,造成難以彌補的後果,從而失去資訊隱藏 (information hiding) 的意義。
因此,我們應該將屬性設成 private ,然後以 getter 與 setter 進行操作哩!
| 中英文術語對照 | |
|---|---|
| 物件 | object |
| 建構子 | constructor |
| 類別 | class |
| 編譯器 | compiler |
| 建構子 | default constructor |
| 關鍵字 | keyword |
| 程式 | program |
| 屬性 | field |
| 參數 | parameter |
| 字面常數 | literal |
| 區域變數 | local variable |
| 資訊隱藏 | information hiding |
您可以繼續參考
軟體開發
相關目錄
回 Java 入門指南
回 Java 教材目錄
回首頁
參考資料
The JavaTM Tutorials: Getting Started
The JavaTM Tutorials: Learning the Java Language
The JavaTM Tutorials: Essential Classes
The Java Language Specification, Third Edition
本文於 2013 年 1 月訂正
4 則留言:
前面的例子還很單純,但是到了本篇,感覺變複雜了
我是初學者,請問是實際需求如此,還是為了要講解「建構子」,才故意改成麻煩的寫法?
public class Encrypt()中有3個public Encrypt() (有的有參數),實在難以理解。
其中有2次執行setArray()
#17. setArray()
前文:其中 a 、 b 、 n 、 c 、 diff ,我們需要提供這五個屬性值,然後呼叫 setArray() 計算出 cArray 的值
→ 但5個變數都尚未給初始值呢,怎麼算?
#38. setArray()的作用又是為何?
不是在EncryptDemo()再叫用t.cArray即可?
謝謝指導!
建構子是一種特別的方法,就像方法可以 overload 一樣,因此建構子可以有多種參數版本,實際建立物件時,可依需要使用某一參數版本的建構子。
其次,建構子視呼叫版本只會執行該一版本的建構子,例如用 Encrypt(5, 7, 33, 'a', 205) ,就會執行具有 5 個參數的建構子。因此執行此一版本的 setArray() 後,其他版本的 setArray() 都不會被執行。
我們的例子是針對英文小寫字母,假設今天要改成將 10 的阿拉伯數字或其他語言的字母編成密碼,相對這些參數值都會不一樣,所以 overload 使 Encrypt() 具有通用性,而非侷限某一特定用途。
最後 Encrypt 的完成品部分,用到了 overload 的概念,
但是這部份似乎沒有提到過,"基礎篇"裡面也沒有,
可能可以在文中稍作說明一下~
另外,this(a, b, 26, 'a', 97) 這種用法,
好像也不曾提及過,不曉得算不算特殊用法?
亦可在文中稍作說明~
this 的用法在新版的入門指南有說明,請參考新版的電子書為準,謝謝 :)
張貼留言