public class FinalTest { public String findword="final word"; public final StringBuffer sfindword=new StringBuffer("final word"); /** * @param args */ public static void main(String[] args) { FinalTest ft=new FinalTest(); System.out.println(ft.findword.hashCode()); ft.findword+="dd"; System.out.println(ft.findword.hashCode()); System.out.println(ft.sfindword.hashCode()); ft.sfindword.append("dd"); System.out.println(ft.sfindword.hashCode()); } }
运行结果为:
-1919389836
-1992659212
14576877
14576877
final修饰符表示只是指向固定的地址,并不能代表是固定的数值
可见,final只对引用的“值”(也即它所指向的那个对象的内存地址)有效,它迫使引用只能指向初始指向的那个对象,改变它的指向会导致编译期错误。至于它所指向的对象的变化,final是不负责的。这很类似==操作符:==操作符只负责引用的“值”相等,至于这个地址所指向的对象内容是否相等,==操作符是不管的。 理解final问题有很重要的含义。许多程序漏洞都基于此----final只能保证引用永远指向固定对象,不能保证那个对象的状态不变。在多线程的操作中,一个对象会被多个线程共享或修改,一个线程对对象无意识的修改可能会导致另一个使用此对象的线程崩溃。一个错误的解决方法就是在此对象新建的时候把它声明为final,意图使得它“永远不变”。其实那是徒劳的。