equals和==
java中的数据类型,可分为两类:
-
1.基本数据类型,也称原始数据类型。byte,short,char,int,long,float,double,boolean
他们之间的比较,应用双等号,比较的是他们的值。基本数据类型比较(string 除外), == 和 Equals 两者都是比较值;
-
2.复合数据类型(类)
当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。
JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地址, 对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值。
但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date,在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。
因为Object的equals方法也是用双等号进行比较的,所以比较后的结果跟双等号(==)的结果相同。
object类中equals源码:
- 一:这里我们来看一段程序:
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = "Monday";
if (s1 == s2)
{
System.out.println("s1 == s2");}
else{
System.out.println("s1 != s2");}
}
}
编译并运行程序,输出:s1 == s2
说明:s1 与 s2 引用同一个 String 对象 – “Monday”!
- 二:在稍微改动以下程序
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
}
我们将s2用new操作符创建
程序输出:
s1 != s2
s1 equals s2
说明:s1 s2分别引用了两个"Monday"String对象
字符串缓冲池
原来,程序在运行的时候会创建一个字符串缓冲池当使用 s2 = “Monday” 这样的表达是创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,在第一个程序中,s1先被放到了池中,所以在s2被创建的时候,程序找到了具有相同值的 s1
将s2引用s1所引用的对象"Monday"
第二段程序中,使用了 new 操作符,他明白的告诉程序:"我要一个新的!不要旧的!"于是一个新的"Monday"Sting对象被创建在内存中。他们的值相同,但是位置不同。
- 三:再次改动程序
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
s2 = s2.intern();
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
}
这次加入:s2 = s2.intern();
程序输出:
s1 == s2
s1 equals s2
原 来,(java.lang.String的intern()方法"abc".intern()方法的返回值还是字符串"abc",表面上看起来好像这个方法没什么用处。但实际上,它做了个小动作:检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会 把"abc"添加到字符串池中,然后再返回它的引用。
hashcode
这里我们在再说一点我遇到的问题。
先上一段代码
static public class Student {
private String name;
private String sex;
public Student(String name, String sex) {
this.name = name;
this.sex = sex;
}
}
基于上面那个student类我们用hashmap
public static void main(String[] args) {
Student stu1 = new Student("Mrlj", "B");
Student stu2 = new Student("Mrlj", "B");
HashMap<Student, String> hMap = new HashMap<>();
System.out.println("--------------put操作----------------");
System.out.println(stu1.hashCode());
System.out.println(stu2.hashCode());
hMap.put(stu1, "1");
hMap.put(stu2, "2");
System.out.println("--------------Get操作----------------");
System.out.println(hMap.get(stu1));
System.out.println(hMap.get(stu2));
}
输出结果位
好的 我们这里修改以下stuent类,就是手动重写了equals和hashcode
static public class Student {
private String name;
private String sex;
public Student(String name, String sex) {
this.name = name;
this.sex = sex;
}
@Override
public boolean equals(Object o) {
if (o instanceof Student) {
Student stu = (Student) o;
System.out.println("this:" + (name + sex) + " " + "that:" + stu.getName() + stu.getSex());
return (name + sex).equalsIgnoreCase(stu.getName().trim() + stu.getSex().trim());
}
return false;
}
@Override
public int hashCode() {
System.out.println("hashCode:"+(name+sex).hashCode());
return (name+sex).hashCode();
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
}
在此写main函数
public static void main(String[] args) {
Student stu1 = new Student("Mrlj", "B");
Student stu2 = new Student("Mrlj", "B");
HashMap<Student, String> hMap = new HashMap<>();
System.out.println("--------------put操作----------------");
hMap.put(stu1, "1");
hMap.put(stu2, "2");
System.out.println("--------------Get操作----------------");
System.out.println(hMap.get(stu1));
System.out.println(hMap.get(stu2));
}
这次输出结果为:
我们发现输出两个都是2!!!我们知道HashMap的put方法当key相同的话就会更新value,不可能有相同的key。这里我们看到这两个因为重写了equals和hashcode所以引用相同所以认为key相同所以将value更新。
前面那个stuent因为没有重写equals和hashcode所以不是同一个引用所以导致直接插入。
那就是在自定义的类中重写hashCode()和equals()阿,定自己的
规则。在HashMap中,equals()是在hashCode()匹配相等之后再使用的,如果
hashCode()匹配不等,程序压根就不调用equals(),这是HashMap源码中定的规则,
在第一个put的时候就调用了hashcode ,然后第二个put时候又调用hashcode之后紧接着自动调用equals所以我们看到输出顺序为