Write a program that makes 2 + 2 = 5,看到这个题目,感觉很新颖,第一个答案就是用Java实现的。用上了Java中的整型实例池的概念。以前只看到过实例池导致两个对象的指针相同的问题,即
1 | Integer a = new Integer( 2 ); |
2 | Integer b = new Integer( 2 ); |
3 | System.out.print(a == b); |
上面的代码最终输出的是true,按照Java对象的申请原则来说,这里应该是false才对。正是因为JVM在实现的时候,默认生成了一些 Integer对象的实例,当需要的实例是池子中已经存在的数值时,直接返回已经生成的对象的引用,不必新构造对象。这样可以极大减少实例数目和程序运行 性能。
而这个题目是将池子中的对象的内容进行了修改,最终使得取回的实例的值发生了改变。这样其实很危险的,如果在正式运行程序的业务代码之前,做这个修改,那么整个程序的运行逻辑将产生混乱。
01 | import java.lang.reflect.Field; |
04 | public static void main(String[] args) throws Exception { |
05 | Class cache = Integer. class .getDeclaredClasses()[ 0 ]; |
06 | Field c = cache.getDeclaredField( "cache" ); |
07 | c.setAccessible( true ); |
08 | Integer[] array = (Integer[]) c.get(cache); |
09 | array[ 132 ] = array[ 133 ]; |
11 | System.out.printf( "%d" , 2 + 2 ); |
上面是具体的代码,最终输出的结果为5,作者给出的解释为:
You need to change it even deeper than you can typically access. Note that this is designed for Java 6 with no funky parameters passed in on the JVM that would otherwise change the IntegerCache. Deep within the Integer class is a Flyweight of Integers. This is an array of Integers from −128 to +127. cache[132] is the spot where 4 would normally be. Set it to 5.
利用缓存的读写接口,将4这个实例的缓存对象的指针改为指向5的实例对象了,这样,当应用程序取出4时,实际上返回的是5的引用,打印出来的也就是5了。
via shentar |