如果一個類中有多個具有相同名稱但引數不同的方法,則稱為方法過載。如果只需要執行一個操作,具有相同的方法名稱將增加程式的可讀性。
假設必須執行給定數值的新增操作(求和),但是引數的數量不固定,如果為兩個引數編寫add1(int,int)
方法,為三個引數編寫add2(int,int,int)
方法,可能對其他程式員來說很難理解這樣的方法的行為,因為它的名稱不同。
因此,執行方法過載能比較容易理解程式含義。
方法過載的優點
過載方法不同的方式
在java中過載方法有兩種方式,它們分別是:
注意:在java中,只通過更改方法的返回型別來實現方法過載是不可以的。
在這個例子中,我們將建立了兩個方法,第一個add()
方法執行兩個數值的相加操作,第二個add()
方法執行三個數值的相加操作。
在這個例子中,我們建立靜態方法,所以不需要建立範例來呼叫方法。
class Adder {
static int add(int a, int b) {
return a + b;
}
static int add(int a, int b, int c) {
return a + b + c;
}
}
class TestOverloading1 {
public static void main(String[] args) {
System.out.println(Adder.add(11, 11));
System.out.println(Adder.add(11, 11, 11));
}
}
上面程式碼執行後,輸出結果如下 -
22
33
在這個例子中,建立了兩種資料型別不同的方法。 第一個add
方法接收兩個整數引數,第二個add
方法接收兩個雙精度型引數。
class Adder {
static int add(int a, int b) {
return a + b;
}
static double add(double a, double b) {
return a + b;
}
}
class TestOverloading2 {
public static void main(String[] args) {
System.out.println(Adder.add(11, 11));
System.out.println(Adder.add(12.3, 12.6));
}
}
上面程式碼執行後,輸出結果如下 -
22
24.9
問題:為什麼方法過載不能通過更改方法的返回型別?
在java中,只通過改變方法的返回型別來實現方法過載是不可能的,因為這樣具有模糊性。 讓我們來看看模糊性是怎麼樣發生的:
class Adder {
static int add(int a, int b) {
return a + b;
}
static double add(int a, int b) {
return a + b;
}
}
class TestOverloading3 {
public static void main(String[] args) {
System.out.println(Adder.add(11, 11));// ambiguity
}
}
上面程式碼執行後,輸出結果如下 -
Compile Time Error: method add(int,int) is already defined in class Adder
System.out.println(Adder.add(11,11));
//這裡,java如何確定應該呼叫哪個sum()
方法?
注意:編譯時錯誤優於執行時錯誤。 所以,如果你宣告相同的方法具有相同的引數,java編譯器渲染編譯器時間錯誤。
可以過載java main()方法嗎?
這是完全可以的。 可以通過方法過載在類中有任意數量的main
方法。 但是JVM呼叫main()
方法,它只接收字串陣列作為引數。 讓我們來看一個簡單的例子:
class TestOverloading4{
public static void main(String[] args){System.out.println("main with String[]");}
public static void main(String args){System.out.println("main with String");}
public static void main(){System.out.println("main without args");}
}
上面程式碼執行後,輸出結果如下 -
main with String[]
如果沒有找到匹配的資料型別,那麼會隱式地將一個型別提升到另一個型別。 讓我們通過下面的圖示來理解這個概念:
如上圖所示,byte
可以被提升為short
,int
,long
,float
或double
。 short
資料型別可以提升為int
,long
,float
或double
。 char
資料型別可以提升為int
,long
,float
或double
等等。
使用型別提升方法過載的範例
class OverloadingCalculation1 {
void sum(int a, long b) {
System.out.println(a + b);
}
void sum(int a, int b, int c) {
System.out.println(a + b + c);
}
public static void main(String args[]) {
OverloadingCalculation1 obj = new OverloadingCalculation1();
obj.sum(20, 20);// now second int literal will be promoted to long
obj.sum(20, 20, 20);
}
}
上面程式碼執行後,輸出結果如下 -
40
60
如果找到匹配使用型別提升方法過載範例
如果方法中存在匹配型別引數,則不會執行型別提升方法過載。
class OverloadingCalculation2 {
void sum(int a, int b) {
System.out.println("int arg method invoked");
}
void sum(long a, long b) {
System.out.println("long arg method invoked");
}
public static void main(String args[]) {
OverloadingCalculation2 obj = new OverloadingCalculation2();
obj.sum(20, 20);// now int arg sum() method gets invoked
}
}
上面程式碼執行後,輸出結果如下 -
Output:int arg method invoked
在模糊的情況下使用型別提升來實現方法過載的範例
如果在方法中沒有匹配的型別引數,並且每個方法都會提升相同數量的引數,那麼會出現歧義。如下範例程式碼,將會產生編譯時錯誤。
class OverloadingCalculation3 {
void sum(int a, long b) {
System.out.println("a method invoked");
}
void sum(long a, int b) {
System.out.println("b method invoked");
}
public static void main(String args[]) {
OverloadingCalculation3 obj = new OverloadingCalculation3();
obj.sum(20, 20);// now ambiguity
}
}
注意: 只有一種型別不被隱式地去提升,例如
double
不能被隱含地提升到任何型別。