一 简单的例子
我们先看一段代码,很简单对不对
执行一下:
和大多数人心里想的不一样吧,好多人还在纠结,到底是全是true还是全是false呀。
然而一个是true,一个是false,这是为什么呢?
有的人根本不去思考,直接打开百度搜索去查找答案(一个个都是百度首席软件工程师)
根据Java编译机制,.java文件在编译以后会生成.class文件给JVM加载执行,于是找到.class文件,反编译看了一下,发现编译器在编译我们的代码时,很调皮(聪明的)的在我们声明的变量加上了valueOf方法 ,代码变成了如下:
看一下valueOf方法的实现
二 反编译以及源码分析
我们发现,Integer的作者在写这个类时,为了避免重复创建对象,对Integer值做了缓存,如果这个值在缓存范围内,直接返回缓存好的对象,否则new一个新的对象返回,那究竟这个缓存到底缓存了哪些内容呢?看一下IntegerCache这个类:
这是一个内部静态类,该类只能在Integer这个类的内部访问,这个类在初始化的时候,会去加载JVM的配置,如果有值,就用配置的值初始化缓存数组,否则就缓存-128到127之间的值。
这里其实也就是运用了jvm里面的常量池,也是JDK8做的优化,这样做有什么好处呢?
仔细思考一个 1-128这些数可能我们在很多代码中都要使用到,JVM底层自己初始化的操作也会用到,虚拟机的本来性能就有限,所以如果我们把这些使用频率很高的一开始就初始化放到常量池里,然后其他的类使用只需要引用其内存地址 不会额外的申请内存是不是也省了一些空间呢。
再来看看我们之前的代码:
看一下IntegerCache这个类:
结论:我们在比较两个Integer对象的值时,无论是怎么声明的,都一定要使用equals去比较,不能用==,在Java中没有重载操作符这一说,特别是从其它语言转到Java的童鞋们要注意。equals我在其它文章里已经做了详细解说。
思考以下代码的执行结果: