美文网首页
逆天改命,Java 反射的黑科技

逆天改命,Java 反射的黑科技

作者: java架构进阶 | 来源:发表于2019-05-24 21:50 被阅读0次

    众所周知,反射是 Java 的一大利器,它可以做到许多看起来不可思议的事情,但是用得不好也会给我们的系统挖下许多坑。下面就介绍一个反射的黑科技,请充分理解并消化里面的知识,并把这项技术用到实际的项目中去

    在开始之前,我们先来念两句诗,代码如下:

    public static void main(String[] args) {
    
        recitePoems(false);
    
        recitePoems(true);
    
    }
    
    private static void recitePoems(Boolean b) {
    
        if (b) {
    
            System.out.println("苟利国家生死以");
    
        } else {
    
            System.out.println("岂因祸福避趋之");
    
        }
    
    }
    

    上面代码的输出是:

    岂因祸福避趋之
    
    苟利国家生死以
    

    不对呀,反了反了,念诗都念错,姿势水平还是太低。怎么改呢,很简单,把两次 recitePoems 方法调用的参数调转过来就可以了? naïve,本文的目的是介绍黑科技,当然不会用这种寻常的办法解决问题。

    不卖关子了,直接上代码吧:

    private static void doSomeMagic() throws Exception {
    
        Field modifiersField = Field.class.getDeclaredField("modifiers");
    
        modifiersField.setAccessible(true);
    
        Field trueField = Boolean.class.getDeclaredField("TRUE");
    
        modifiersField.set(trueField, trueField.getModifiers() & ~Modifier.FINAL);
    
        Field falseField = Boolean.class.getDeclaredField("FALSE");
    
        modifiersField.set(falseField, falseField.getModifiers() & ~Modifier.FINAL);
    
        Boolean trueValue = true;
    
        trueField.set(null, false);
    
        falseField.set(null, trueValue);
    
    }
    

    接下来,只需要在 main 方法的开头调用这个名为 doSomeMagic膜法方法就好了:

    public static void main(String[] args) throws Exception {
    
        doSomeMagic();
    
        recitePoems(false);
    
        recitePoems(true);
    
    }
    

    修改完毕之后,我们得到了期望的输出:

    苟利国家生死以
    
    岂因祸福避趋之
    

    那么,doSomeMagic 方法到底干了什么呢?很简单,它交换了 Boolean.TRUEBoolean.FALSE 的值。为了能够重写它们的值,我们需要去掉它们的 final 修饰符,这就是 xxxField.getModifiers() & ~Modifier.FINAL 的作用。

    交换 Boolean.TRUEBoolean.FALSE的值,为什么能够改变原代码的运行逻辑呢?我们看到,recitePoems 方法的形参是 boolean 的包装类型 Boolean,直接将 truefalse 作为实参调用它时,将会发生自动装箱操作。而自动装箱操作是通过调用 Boolean.valueOf() 方法完成的,我们看看这个方法的源码:

    /**
    
     * Returns a {@code Boolean} instance representing the specified
    
     * {@code boolean} value.  If the specified {@code boolean} value
    
     * is {@code true}, this method returns {@code Boolean.TRUE};
    
     * if it is {@code false}, this method returns {@code Boolean.FALSE}.
    
     * If a new {@code Boolean} instance is not required, this method
    
     * should generally be used in preference to the constructor
    
     * {@link #Boolean(boolean)}, as this method is likely to yield
    
     * significantly better space and time performance.
    
     *
    
     * @param  b a boolean value.
    
     * @return a {@code Boolean} instance representing {@code b}.
    
     * @since  1.4
    
     */
    
    public static Boolean valueOf(boolean b) {
    
        return (b ? TRUE : FALSE);
    
    }
    

    可以看到,Boolean.valueOf() 方法直接使用了 Boolean.TRUEBoolean.FALSE 两个常量。这就是我们能做到如此“是非颠倒”的原因。

    所以说,一个程序的命运啊,当然要靠自我的奋斗,但也要考虑历史的进程。你绝对不会知道,好好的一个 true,怎么就变成 false 了呢。

    这篇文章讲了这么久也没别的,大概三件事:一个,去掉 Boolean.TRUEBoolean.FALSE 的 final 修饰符;第二个,交换了它们的值;第三个,就是基本类型自动装箱的细节;如果说还有一点成绩,那就是在公司每个项目的 main 方法上调用了一下 doSomeMagic 方法,这对于被炒鱿鱼的命运有很大的关系。

    很惭愧,就做了一点微小的工作,谢谢大家。

    相关文章

      网友评论

          本文标题:逆天改命,Java 反射的黑科技

          本文链接:https://www.haomeiwen.com/subject/ibqozqtx.html