反射引用 private 屬性,并修改 static、static final 修飾的域

static、final、static final 引用

本人見識有限,若有問題歡迎拍磚、學習。此外這里主要是我發(fā)現(xiàn)一個問題,想讓大家都看看,討論討論。

背景:之前看見項目里面有這么寫單例的:

public class A {

    private String name;

    private static  A a = new A();

    private A() {

    }

    public static  A factoryA() {
        return a;
    }
}
// omit getter and setter for name property

看到這段代碼我就想到,之前自己都是一直用 static final 修飾的。通過 static final 定義的引用 可作為單例使用。這里,應該牽扯 2 個概念有必要提一下:

  1. static 修飾的引用存儲在堆中,會被線程共享。
  2. final 修飾的引用不能指向其他對象(這個我有個疑問,所以才有了這篇文章)。

所以,線程安全的對象我就理解為單例了。(我潛意識里面,認為不能修改引用的對象就為單例,顯然不對。額~ 感覺沒描述清楚。。。)

回到上面的代碼,沒有用 final 修飾。如果 引用 a 指向了其他對象,那通過 factoryA() 獲取的方法就不能保證在系統(tǒng)中是同一個對象。所以我就在想怎么修改,第一個想到的就是反射,然后有了下面的代碼:

public static void main(String[] args) throws Exception {
        A a = A.factoryA();
        A b = A.factoryA();
        A c = null;

        a.setName("not be replace");
        Class<? extends A> clazz = A.class;

        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            field.setAccessible(true);
            System.out.println(field.getName() + "\t" + field.get(a).getClass().getSimpleName());
            if (field.getName().equals("a")) {
                field.set(null, new A("you have been replace"));
                c = (A) field.get(null);
            }
        }

        A e = A.factoryA();

        System.out.println("a:\t" + a.getName());
        System.out.println("b:\t" + b.getName());
        System.out.println("c:\t" + c.getName());
        System.out.println("e:\t" + e.getName());

    }//:~out
    /**
     * name String 
     * a    A
     * a:   not be replace
     * b:   not be replace
     * c:   you have been replace
     * e:   you have been replace 
     *
     */

結(jié)果很顯然,static 修飾的域被修改了。(我也不確定,是不是代碼的問題或我對放射理解的問題,但是這里看到的結(jié)果就是這樣。)

到這兒,我在 static 修飾符后面添加了 final 關鍵字。

    private static final  A a = new A();

如此,上面的代碼顯然不能正常干活了。但是在 stackoverflow上找到了方法(這段代碼是網(wǎng)上的):

     static void setFinalStatic(Field field, Object newValue) throws Exception {
            field.setAccessible(true);
            Field modifiersField = Field.class.getDeclaredField("modifiers");

            // wrapping setAccessible 
            AccessController.doPrivileged(new PrivilegedAction() {
                @Override
                public Object run() {
                    modifiersField.setAccessible(true);
                    return null;
                }
            });

            modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
            field.set(null, newValue);
        }

如下調(diào)用后:

    setFinalStatic(A.class.getDeclaredField("a"), new A("you have been replace"));
        
        A e = A.factoryA();

        System.out.println("a:\t" + a.getName());
        System.out.println("b:\t" + b.getName());
    //  System.out.println("c:\t" + c.getName());
        System.out.println("e:\t" + e.getName());

結(jié)果表明,static final 修飾的引用在上面的方式下會被指向其他對象。所以,我的疑問就是:不是說 final 修飾引用不能指向其他對象,那么上面的試驗結(jié)果怎么解釋……

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內(nèi)部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,894評論 18 399
  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時...
    歐辰_OSR閱讀 30,286評論 8 265
  • 春臨天姥綠遮坡,鳥筑云巢花放歌。 歲末重行磕山路,殘枝錯落少風過。
    夢之旅_926e閱讀 191評論 9 7
  • 2018年個人成長五大目標 目標一、標準化持續(xù)落地,打造標準化體系和監(jiān)督體系。 目標二、專業(yè)維修技術加強...
    凌勝亮閱讀 130評論 0 0

友情鏈接更多精彩內(nèi)容