首先上案例:
public static void main(String[] args){
String str1 = "abc";
String str2 = "abc";
String str3 = new String("abc");//new出来的新地址
System.out.println(5==5);//基本数据类型比较值
System.out.println(str1==str2);//比较的是地址 跟重写前的equals的方式一样//true
System.out.println(str1.equals(str2));//比较的是值
System.out.println(str1==str3);//比较的是地址 //flase
System.out.println(str1.equals(str3));//比较的是值
从上述中我们可以看到:基本数据类型5==5
是比较值,注意这里是基本数据类型
要从String区分:
因为String是引用类型引用类型比较的是地址
有人问了:那str1==str2
肯定是false //错
其实应该是true,下面来看为什么 由于我们java虚拟机会讲定义的变量和存放的位置 区
分,这和c中的指针差不多 ,java就有了引用这一说 下面看一下为什么:
JVM为了提高性能和减少内存开销,在实例化字符串常量的时候进行了一些优化
为字符串开辟一个字符串常量池,类似于缓存区
创建字符串常量时,首先坚持字符串常量池是否存在该字符串
存在该字符串,返回引用实例,不存在,实例化该字符串并放入池中
所以大家肯定会说str1==str2是相同的地址,那为什么str1==str3
肯定也是true?
答案是 false因为 是新New的空间地址,看下图:
好了 我们再说equals比较,由于String类已经重写了equals 为了避免废话我们直接上源码:
看不懂没关系我们下面用实体类进行演示重写:
实体类案例:
public class Person {
private String name;
private int age;
//无参构造
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
//重写equals方法
@Override
public boolean equals(Object obj) {
//判断内存地址
if (obj == this) {
return true;
}
if (obj == null) {
return false;
}
//判断是否是同一类型的对象
if (obj instanceof Person) {
//强制转换成Person类型
Person per = (Person) obj;
//判断他们的属性值 注:这里的age为什么要用==?可以在评论区回答
if (this.name.equals(per.name) && this.age == (per.age)) {
return true;
}
}
return false;
}
}
最后总结:
基本数据类型比较的是值
引用数据类型”:
比较的是地址
想要比较引用类型的值
就重写equals
比如没重写前==等同于equals,都是比较的地址
重写后前者是比较地址,后者是比较值