1. 程式人生 > >Java - 21 Java 重寫(Override)與過載(Overload)

Java - 21 Java 重寫(Override)與過載(Overload)

Java 重寫(Override)與過載(Overload)


重寫(Override)

重寫是子類對父類的允許訪問的方法的實現過程進行重新編寫!返回值和形參都不能改變。即外殼不變,核心重寫!

重寫的好處在於子類可以根據需要,定義特定於自己的行為。

也就是說子類能夠根據需要實現父類的方法。

在面向物件原則裡,重寫意味著可以重寫任何現有方法。例項如下:

class Animal{

   public void move(){
      System.out.println("動物可以移動");
   }
}

class Dog extends Animal{

   public void move(){
      System.out.println("狗可以跑和走");
   }
}

public class TestDog{

   public static void main(String args[]){
      Animal a = new Animal(); // Animal 物件
      Animal b = new Dog(); // Dog 物件

      a.move();// 執行 Animal 類的方法

      b.move();//執行 Dog 類的方法
   }
}

以上例項編譯執行結果如下:

動物可以移動
狗可以跑和走

在上面的例子中可以看到,儘管b屬於Animal型別,但是它執行的是Dog類的move方法。

這是由於在編譯階段,只是檢查引數的引用型別。

然而在執行時,Java虛擬機器(JVM)指定物件的型別並且執行該物件的方法。

因此在上面的例子中,之所以能編譯成功,是因為Animal類中存在move方法,然而執行時,執行的是特定物件的方法。

思考以下例子:

class Animal{

   public void move(){
      System.out.println("動物可以移動");
   }
}

class Dog extends Animal{

   public void move(){
      System.out.println("狗可以跑和走");
   }
   public void bark(){
      System.out.println("狗可以吠叫");
   }
}

public class TestDog{

   public static void main(String args[]){
      Animal a = new Animal(); // Animal 物件
      Animal b = new Dog(); // Dog 物件

      a.move();// 執行 Animal 類的方法
      b.move();//執行 Dog 類的方法
      a.bark();//執行 Animal 類的方法
   }
}

以上例項編譯執行結果如下:

TestDog.java:30: cannot find symbol
symbol  : method bark()
location: class Animal
                a.bark();
                 ^

該程式將丟擲一個編譯錯誤,因為a的引用型別Animal沒有bark方法。


方法重寫的規則

  • 引數列表必須完全與被重寫方法的相同;
  • 返回型別必須完全與被重寫方法的返回型別相同;
  • 子類方法的訪問許可權必須大於或等於父類方法的訪問許可權。例如:如果父類的一個方法被宣告為public,那麼在子類中重寫該方法就不能宣告為protected。
  • 父類的成員方法只能被它的子類重寫。
  • 宣告為final的方法不能被重寫。
  • 宣告為static的方法不能被重寫,但是能夠被再次宣告。
  • 如果一個方法不能被繼承,那麼該方法不能被重寫。
  • 子類和父類在同一個包中,那麼子類可以重寫父類所有方法,除了宣告為private和final的方法。
  • 子類和父類不在同一個包中,那麼子類只能夠重寫父類的宣告為public和protected的非final方法。
  • 重寫的方法能夠丟擲任何非強制異常,無論被重寫的方法是否丟擲異常。但是,重寫的方法不能丟擲新的強制性異常,或者比被重寫方法宣告的更廣泛的強制性異常,反之則可以。
  • 構造方法不能被重寫。
  • 如果不能繼承一個方法,則不能重寫這個方法。

Super關鍵字的使用

當需要在子類中呼叫父類的被重寫方法時,要使用super關鍵字。

class Animal{

   public void move(){
      System.out.println("動物可以移動");
   }
}

class Dog extends Animal{

   public void move(){
      super.move(); // 應用super類的方法
      System.out.println("狗可以跑和走");
   }
}

public class TestDog{

   public static void main(String args[]){

      Animal b = new Dog(); //
      b.move(); //執行 Dog類的方法

   }
}

以上例項編譯執行結果如下:

動物可以移動
狗可以跑和走

過載(Overload)

過載(overloading) 是在一個類裡面,方法名字相同,而引數不同。返回型別呢?可以相同也可以不同。

每個過載的方法(或者建構函式)都必須有一個獨一無二的引數型別列表。

只能過載建構函式

過載規則

  • 被過載的方法必須改變引數列表;
  • 被過載的方法可以改變返回型別;
  • 被過載的方法可以改變訪問修飾符;
  • 被過載的方法可以宣告新的或更廣的檢查異常;
  • 方法能夠在同一個類中或者在一個子類中被過載。

例項

public class Overloading {
 
	public int test(){
		System.out.println("test1");
		return 1;
	}
 
	public void test(int a){
		System.out.println("test2");
	}	
 
	//以下兩個引數型別順序不同
	public String test(int a,String s){
		System.out.println("test3");
		return "returntest3";
	}	
 
	public String test(String s,int a){
		System.out.println("test4");
		return "returntest4";
	}	
 
	public static void main(String[] args){
		Overloading o = new Overloading();
		System.out.println(o.test());
		o.test(1);
		System.out.println(o.test(1,"test3"));
		System.out.println(o.test("test4",1));
	}

重寫與過載之間的區別

區別點 過載方法 重寫方法
引數列表 必須修改 一定不能修改
返回型別 可以修改 一定不能修改
異常 可以修改 可以減少或刪除,一定不能丟擲新的或者更廣的異常
訪問 可以修改 一定不能做更嚴格的限制(可以降低限制)