在Java中需要判断两个变量相等有两种方法 == 和equals
说一下两者的区别:
‘==’比较的是两个变量的内存地址。而equals比较的是两个变量的值。
说一个比较经典的问题
String a = "111"; String b = "111"; Systom.out.printlen(a == b);
这个打印结果是 :“true”
而equals的结果也是true,刚开始我也不知道问什么。
然后就出现了下边这些代码
public static void main(String[] args) { String a = "111"; String b = "111"; String c = new String("111"); System.out.println("--获取内存地址--"); System.out.println(VM.current().addressOf(a)); System.out.println(VM.current().addressOf(b)); System.out.println(VM.current().addressOf(c)); }
我发现 a 和 b 的内存地址居然是一样的
于是上网查了下才知道是jvm虚拟机中的常量池
在JVM中有一块区域叫做常量池,常量池中的数据是那些在编译期间被确定的,并被保存在已编译的字节码(编译后产生的class文件)文件中的一些数据。Java栈的特点是存取速度快(比堆快),但是空间小,数据生命周期固定,只能生存到方法结束。我们定义的String a = “111”; String b = “111”; 这些语句,我们拆分开来看:
-
a = “111”,等号右边的指的是编译期间可以被确定的内容,都维护在常量池中。
-
b = “111” ,等号左边的指的是一个引用,引用的内容是等号右边数据在常量池中的地址
在这时由于方法没有结束,所以他们在常量池中的地址是一样的 而这里也涉及到 =(赋值符号)的用法,当我们 a="123"时,并不是把123的值给了a 而是把123 在内存中的地址给了a(类似于c语言的指针)是a指向了存储 123 值的地址,所以这个时候 == 和 equals 的结果是一样的
而当我们 new 了一个对象的时候,jvm 就会 分配一个新的地址,即使你的值在常量池中已经存在了
public static void main(String[] args) { String a = "111"; String b = "111"; String c = new String("111"); System.out.println("--获取内存地址--"); System.out.println(VM.current().addressOf(a)); System.out.println(VM.current().addressOf(b)); System.out.println(VM.current().addressOf(c)); System.out.println(a == c); System.out.println(a.equals(c)); }
这时候,我们就发现了 == 和 equals 的区别了
下面附上获取内存地址的方法
maven
org.openjdk.jol jol-core0.9
jar 包
找到jar包
jol-core
jar包的话上传不了资源在csdn 里已经存在了如果有需要的可以去网盘
链接:
https://pan.baidu.com/s/1p78J14ApIxl8RY8TbHXQiw
提取码: le1m
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)