Java中final修飾的方法是否可以被重寫
- 2020 年 11 月 3 日
- 筆記
這是一次阿裏面試里被問到的題目,在我的印象中,final修飾的方法是不能被子類重寫的。如果在子類中重寫final修飾的方法,在編譯階段就會提示Error。但是回答的時候還是有點心虛的,因為final變量就可以用反射的方法進行修改,我也不太確定是否有類似的機制可以繞過編譯器的限制。於是面試之後特地上網搜了下這個問題,這裡簡單記錄一下。
首先說一下結論:沒有辦法能夠做到重寫一個final修飾的方法,但是有其他的方法可以接近在子類中重新實現final方法並在運行時的動態綁定的效果。
這裡需要用到一個aop框架叫aspectj,它和spring aop都是比較常用的aop框架。區別是spring aop是基於動態代理的,而aspectj有獨立的編譯器可以實現靜態代理。關於aspectj的安裝配置網上有很多文章了,這裡就不再贅述,直接快進到例子。
首先定義一個SuperClass並在其中定義一個final方法。
SuperClass.java
public class SuperClass {
public final void doSomething() {
System.out.println("super class do something");
}
public static void main(String[] args) {
SuperClass instance = new SubClass(); //此處是父類引用和子類對象
instance.doSomething();
}
}
SubClass.java
public class SubClass extends SuperClass {
//doSomething是final方法,無法被重寫
}
super class do something
Process finished with exit code 0
運行main方法,SubClass繼承了doSomething方法,但是不能重寫,所以通常情況下調用的一定是SuperClass的doSomething方法。
在SubClass中實現「重寫」的doSomething方法
SubClass.java
public class SubClass extends SuperClass {
//doSomething是final方法,無法被重寫
//子類只能在另一個函數中實現重寫的邏輯
protected void overrideDoSomething() {
System.out.println("sub class do something");
}
}
利用環繞通知修改實際調用的方法
DoSomethingAspect.aj
public aspect DoSomethingAsepct {
// 環繞通知 匹配SuperClass類的doSomething方法
void around() : execution(* SuperClass.doSomething()) {
if (thisJoinPoint.getThis() instanceof SubClass) {
//調用子類方法
((SubClass)thisJoinPoint.getThis()).overrideDoSomething();
} else {
//調用原方法
proceed();
}
}
}
運行結果
sub class do something
Process finished with exit code 0
可以看到,調用SubClass的doSomething方法時實際調用的是SubClass類的overrideDoSomething方法,而如果是SuperClass對象的話調用的又是SuperClass里的doSomething方法。根據實際的類型決定調用的方法,就比較接近動態綁定的機制了。而僅從調用的代碼來看和子類重寫方法(雖然實際是final)的效果是一樣的。