首页 > 编程语言 >深入理解 Java 的整型类型:如何实现 2+2=5?

深入理解 Java 的整型类型:如何实现 2+2=5?

时间:2023-04-16 17:11:32浏览次数:35  
标签:缓存 Java cache 深入 整型 数组 Integer array

在开始关于 Java 的整型类型讨论之前,让我们先看下这段神奇的Java代码:

public static void main(String[] args) throws Exception {
      doSomethingMagic();
      System.out.printf("2 + 2 = %d", 2 + 2);
}

执行结果,控制台打印的内容:2 + 2 = 5

那么 doSomethingMagic 方法到底做了什么神奇的事情呢?

private static void doSomethingMagic() throws Exception {
   Class cache = Integer.class.getDeclaredClasses()[0];
   Field c = cache.getDeclaredField("cache");
   c.setAccessible(true);
   Integer[] array = (Integer[]) c.get(cache);
   array[132] = array[133];
}

所以这个例子其实包含了 Java 中整型类型 Integer 的一个知识点。

可能有的朋友对于 doSomethingMagic 里面的代码有点摸不着头脑,让我们先查看上图第17行 2 + 2 反编译出来的代码:

编译器将 2 + 2 的值先计算出来,等于 4.

最后 System.out.println 打印出来的值,实际上是 Integer.valueOf(4) 表达式的返回值。

那么我们进一步查看 JDK 里 Integer.valueOf 的实现:

上面的实现代码,从 830 行到 832行,逻辑是:如果 valueOf 方法调用的参数 i 在 IntegerCache.low 和 IntegerCache.high 之间,即位于 [-128, 127] 的闭区间,则直接从 IntegerCache 这个内部类的缓存数组里取出元素,作为方法的返回值。

只有当输入参数 i 不在 [-128,127] 区间内,才执行代码第 832 行,基于输入参数 i 创建一个新的 Integer 实例。

带着这个理念,我们再看 doSomethingMagic 的代码就清楚多了。

这段代码的作用是访问 Java Integer 类实现的某个内部类维护的缓存数组,对其中一个元素的值进行修改。具体解释如下:

  • Class cache = Integer.class.getDeclaredClasses()[0]:通过 Integer.class.getDeclaredClasses() 方法获取 Integer 类的所有内部类,然后 [0] 取出第一个内部类,也就是 Integer$IntegerCache 类的 Class 对象,并将其赋值给变量 cache. Integer$IntegerCache 类是 Java 8 引入的一个内部类,它用于缓存整数对象(Integer 类型的对象),默认缓存范围是 -128 到 127。

  • Field c = cache.getDeclaredField("cache"): 接下来,通过 cache.getDeclaredField("cache") 方法获取 cache 字段的 Field 对象,并将其赋值给变量 c。该字段存储了 Integer 类的缓存数组。

  • c.setAccessible(true):通过 c.setAccessible(true) 方法设置 c 字段的访问权限,以便可以通过反射来修改该字段的值。

  • Integer[] array = (Integer[]) c.get(cache):通过 c.get(cache) 方法获取缓存数组的值,并将其强制转换为 Integer 类型的数组,然后将其赋值给变量 array.

  • array[132] = array[133]:最后,将缓存数组中下标为 133 的元素的值赋给下标为 132 的元素,从而修改了缓存数组中下标为 132 的元素的值。

我们从 Eclipse 调试器里发现,Integer 内部类的 cache 数组里,第 132 个元素的值为 4,第 133 个元素的值为 5. 本来 Integer.valueOf 方法,对于输入 4,从Integer 内部类缓存数组里,返回第 132 个元素的值,即 4.

现在这个元素的值被第 133 个元素即 5 覆盖了,所以最后得到了 2 + 2 = 5。

用一句话概括这个场景:2 + 2 = 4 = Integer.valueOf(4) = 5 ( 因为 4 在 Integer 内部类 cache 数组里对应的记录,已经被我们的 doSomethingMagic 方法,显式从 4 替换成了 5)。

这就是 Java 里实现 2 + 2 = 5 这个算式的方法之一。

标签:缓存,Java,cache,深入,整型,数组,Integer,array
From: https://www.cnblogs.com/sap-jerry/p/17323601.html

相关文章

  • js 传递汉字 乱码_JavaScript 字符串反转乱码问题解决
    https://blog.csdn.net/weixin_36483301/article/details/113451892emoji表情和非常用字实际解决中文编码问题,可以通过解码解决js中使用decodeURL即可解决......
  • java反射
    java反射1.基本定义Java反射机制是在运行状态时,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有的方法和属性),这种动态获取的信息以及动态调用对象的方法的功能就称为java语言的反射机制。可以说为对象可以通过......
  • Java中abstract(抽象类)
    1、概述(1)只给出方法定义而不具体实现的方法被称为抽象方法,抽象方法是没有方法体的,在代码的表达上就是没有“{}”。使用abstract修饰符来表示抽象方法和抽象类。(2)abstract修饰符表示所修饰的类没有完全实现,还不能实例化。如果在类的方法声明中使用abstract修饰符,表明该方法是一个......
  • Java JDBC批处理添加出现问题,求解决方案
    晚辈使用JDBC批处理时出现一个问题,使用addBatch()方法将记录加入批处理中,我想让这五千条记录每达到一千条记录再执行,以此提高效率,可最后执行在数据库查看时仅五条记录,我尝试将 preparedStatement.executeUpdate();提出if语句,虽然是有五千条记录,但效率相当的慢请求前辈们给出解决......
  • Android深入学习之LayoutInflater类和ViewBinding
    在build.gradle(Module)中添加viewBinding元素后,Android会自动给模块中的每个XML布局文件生成一个相应的Binding类,该Binding类名称为XML布局文件驼峰式大写+Binding后缀。以如下所示的activity_welcome.xml文件为例,对应的ActivityWelcomeBinding.java的源代码如下所示。<?xmlv......
  • java多版本共存
    原理通过脚步改变path环境变量来实现java多版本切换.这里使用的是Win10.一,删除原有的java搜索路径.在安装高版本的java时,会添加一个路径到path环境变量中,如我的C:\ProgramFiles\CommonFiles\Oracle\Java\javapath,在该目录下存有java.exe和javac.exe等.在用cmd执行命令时,......
  • java语句
    语句语句是以;或}或)结尾的一段代码,目的是执行某些操作,并且没有返回值。语句块和方法体也算语句。所以,语句是可以嵌套的。函数体是一个有名称的语句。对于拥有子语句的语句,称为复合语句,否则就是简单语句。 简单语句空语句:;;;方法调用语句:setVal(34);表达式语句:i......
  • javascript 把嵌套的 map 转成 object,再转 json 字符串
    使用JSON.stringify转map时发现并没有转成想要的JSON数据,搜索发现要转成Object才能够转成完整的JSON,用递归转换:constconvertNestedMapToObject=(map)=>{if(mapinstanceofMap){map.forEach((v,k)=>{......
  • javascript常用的循环对比及性能分析
    结论:js中的for循环只有在处理百万条数据以上才会展示出他的强大性能,和看出明显优势,但是在百万条数据往下甚至到个位数的数据量通常都是for和while还有do...while不相上下,反而后两者更加优势明显下面是测试耗时截图(在不同浏览器也会有所不同,我这是Chrome版本111.0.5563.149)......
  • 使用文本编辑器和jdk开发简单JavaSE工程
    一个在eclipse下简单的示例 运行时可以在代码编辑器页面右键run测试发布时可以file->export->runnablejarfile 如果没有eclipse只有jdk和文本编辑器呢? 创建和eclipse工程相似的目录 目录JAVASE01binlibsrc\com\zt\javase01\IODemo.java 下面的命令是在编译class......