final的作用
final根據(jù)修飾位置的不同作用也不相同章母,針對三種情況:
1)修飾變量母蛛,被final修飾的變量必須要初始化,賦初值后不能再重新賦值乳怎。
注意:局部變量不在我們討論的范疇彩郊,因為局部變量本身就有作用范圍前弯,不使用private、public等詞修飾秫逝。
2)修飾方法恕出,被final修飾的方法代表不能重寫。
3)修飾類违帆,被final修飾的類浙巫,不能夠被繼承。
注意:final修飾的類刷后,類中的所有成員方法都被隱式地指定為final方法的畴。
所有的final修飾的字段都是編譯期常量嗎?
并不是尝胆,比如String
如何理解private所修飾的方法是隱式的final丧裁?
不太正確,private可以通過反射的方式去修改,final如果通過反射的方式去修改值的話含衔,必須要避開編譯的自動優(yōu)化煎娇,例如這樣:private final int a=(null!=null?0:1);
Java反射機制可以動態(tài)修改實例中final修飾的成員變量嗎?
回答是分兩種情況的贪染。
當(dāng)final修飾的成員變量在定義的時候就初始化了值缓呛,那么java反射機制就已經(jīng)不能動態(tài)修改它的值了。
當(dāng)final修飾的成員變量在定義的時候并沒有初始化值的話抑进,那么就還能通過java反射機制來動態(tài)修改它的值强经。
public class MainTest {
public static void main(String[] args) {
try {
A a = new A();
Field field = a.getClass().getDeclaredField("a");
field.setAccessible(true);
field.set(a, 2);
System.out.println(field.get(a)); //打印 2
System.out.println(a.getA()); //打印 1 為什么這里不是2
} catch (SecurityException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
}
class A {
private final int a=1 ;
public int getA()
{
return this.a;
}
}
如果把A修改為
class A {
private final int a ; //區(qū)別在于這里沒給a直接賦值,而是用構(gòu)造函數(shù)
public A()
{
this.a=1;
}
public int getA()
{
return this.a;
}
}
System.out.println(field.get(a)); //打印 2
System.out.println(a.getA()); //打印 2 為什么這次是2了
原因在于編譯期間final類型的數(shù)據(jù)自動被優(yōu)化了寺渗,即:所有用到該變量的地方都唄替換成了常量匿情。所以 get方法在編譯后自動優(yōu)化成了return 1 ; 而不是 return this.a信殊;
要想不被自動優(yōu)化炬称,可以把初始化代碼改成:private final int a=(null!=null?0:1);