java中equals和==

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所以我们看到输出顺序为
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值