对java中的equals()方法的总结
Java的基础学习总结--equals方法
一,等于方法介绍
1.1。通过下面的例子掌握等于的用法
1 package cn.galc.test;
2
3 public class TestEquals {
4 public static void main(String[] args) {
5 /**
6 * 这里使用构造方法Cat()在堆内存里面new出了两只猫,
7 * 这两只猫的color,weight,height都是一样的,
8 * 但c1和c2却永远不会相等,这是因为c1和c2分别为堆内存里面两只猫的引用对象,
9 * 里面装着可以找到这两只猫的地址,但由于两只猫在堆内存里面存储在两个不同的空间里面,
10 * 所以c1和c2分别装着不同的地址,因此c1和c2永远不会相等。
11 */
12 Cat c1 = new Cat(1, 1, 1);
13 Cat c2 = new Cat(1, 1, 1);
14 System.out.println("c1==c2的结果是:"+(c1==c2));//false
15 System.out.println("c1.equals(c2)的结果是:"+c1.equals(c2));//false
16 }
17 }
18
19 class Cat {
20 int color, weight, height;
21
22 public Cat(int color, int weight, int height) {
23 this.color = color;
24 this.weight = weight;
25 this.height = height;
26 }
27 }
1.2。画出内存分析图分析C1和C2比较的结果
程序:
Cat c1 =新Cat(1,1,1);
Cat c2 =新Cat(1,1,1);
执行完之后内存之中的布局如下图所示,
C1指向一个对象,C2也指向一个对象,C1和C2里面装着的是这两只猫对象在堆内存里面存储的地址,由于这两只猫对象分别位于不同的存储空间,因此C1和C2里面装着的地址肯定不相等,因此C1和C2这两个引用对象也肯定不相等因此执行:“的System.out.println(C1 == C2);”打印出来的结果肯定是假因此你新出来了两个对象,你放心,这两个对象的引用永远不一样,一样的话就会把其中一个给覆盖掉了,这个可不成.c1是不是等于C2比较的是C1和C2这两个引用里面装着的内容,因为新出来的两个对象的它们的引用永远不一样,因此C1和C2这两个引用的内容也永远不一样,因此C1永远不可能等于C2。因此通过比较两个对象的引用是永远无法使得两个对象相等的,一模一样的。
要想判断两个对象是否相等,不能通过比较两个对象的引用是否相等,这是永远都得不到相等的结果的,因为两个对象的引用永远不会相等,所以正确的比较方法是直接比较这两个对象,比较这两个对象的实质是不是一样的,即这两个对象里面的内容是不是相同的,通过比较这两个对象的属性值是否相同而决定这两个对象是否相等。
对象类提供了一个等于()方法来比较两个对象的内容是否相同,因此我们可以采用这个方法去比较两个对象是否在逻辑上“相等”如:c1.equals(C2);这里是调用从对象类继承下来的的equals()方法,通过查阅API文档得到对象类里的等于方法的定义如下:
public boolean equals(Object obj)
在对象这个类里面提供的等于()方法默认的实现是比较当前对象的引用和你要比较的那个引用它们指向的是否是同一个对象,即和“的C1 == C2”这种写法是一样的“c1.equals(C2)”与“C1 == C2”是完全等价的。因此直接使用继承下来的的equals()方法也是无法直接比较两个对象的内容是否相同的,为此,我们必须得重写的equals()方法,改变这个方法默认的实现。
下面在猫里面类重写这个继承下来的equals()方法方法:
1 class Cat {
2 int color, weight, height;
3
4 public Cat(int color, int weight, int height) {
5 this.color = color;
6 this.weight = weight;
7 this.height = height;
8 }
9
10 /**
11 * 这里是重写相等从Object类继承下来的equals()方法,改变这个方法默认的实现,
12 * 通过我们自己定义的实现来判断决定两个对象在逻辑上是否相等。
13 * 这里我们定义如果两只猫的color,weight,height都相同,
14 * 那么我们就认为这两只猫在逻辑上是一模一样的,即这两只猫是“相等”的。
15 */
16 public boolean equals(Object obj){
17 if (obj==null){
18 return false;
19 }
20 else{
21 /**
22 * instanceof是对象运算符。
23 * 对象运算符用来测定一个对象是否属于某个指定类或指定的子类的实例。
24 * 对象运算符是一个组合单词instanceof。
25 * 该运算符是一个双目运算符,其左边的表达式是一个对象,右边的表达式是一个类,
26 * 如果左边的对象是右边的类创建的对象,则运算结果为true,否则为false。
27 */
28 if (obj instanceof Cat){
29 Cat c = (Cat)obj;
30 if (c.color==this.color && c.weight==this.weight && c.height==this.height){
31 return true;
32 }
33 }
34 }
35 return false;
36 }
37 }
此时在再主方法里面执行打印的命令:
1 public static void main(String[] args) {
2 /**
3 * 这里使用构造方法Cat()在堆内存里面new出了两只猫,
4 * 这两只猫的color,weight,height都是一样的,
5 * 但c1和c2却永远不会相等,这是因为c1和c2分别为堆内存里面两只猫的引用对象,
6 * 里面装着可以找到这两只猫的地址,但由于两只猫在堆内存里面存储在两个不同的空间里面,
7 * 所以c1和c2分别装着不同的地址,因此c1和c2永远不会相等。
8 */
9 Cat c1 = new Cat(1, 1, 1);
10 Cat c2 = new Cat(1, 1, 1);
11 System.out.println("c1==c2的结果是:"+(c1==c2));//false
12 System.out.println("c1.equals(c2)的结果是:"+c1.equals(c2));//true
13 }
这一次得到的结果就与上次没有重写的equals()方法时得到的结果就不一样了:
“System.out.println(c1 == c2);”打印出来的结果依然是false,因为这里是比较两个对象的引用里面的内容,这两个引用里面的内容当然不相等,而且永远不会相等,所以打印出来的结果肯定是假的。
“System.out.println(c1.equals(c2));”打印出来的结果为真,因为我们在Cat类里面重写了equals()方法,改变了这个方法默认的实现,我们把方法的实现改为只要这个两个对象是真的存在,并且都是猫,并且它们的颜色(颜色),身高(高度)和体重(重量)都相同,那么这两只猫在逻辑上就是一模一样的,是完全相同的两只猫,即这两只猫是“相等”的。所以这里打印出来的结果是正确的。
1.3。如何比较两个字符串对象是否相等?
看下面的例子:
1 public class TestEquals {
2
3 public static void main(String args[]){
4 String s1 = new String("hello");
5 String s2 = new String("hello");
6 System.out.println("s1 == s2的结果是:"+(s1 == s2));//false
7 System.out.println("s1.equals(s2)的结果是:"+s1.equals(s2));//true
8 }
9 }
这一次是比较两个字符串对象是否相等:
System.out.println(s1 == s2);
打印出来的结果依然是FASE,因为这里比较的是S1和S2两个字符串对象的引用,两个对象的引用永远不会相等,所以打印出来的结果为假。
的System.out.println(s1.equals(S2));
打印出来的结果为真实的,因为在字符串类里面重写了从对象类继承(所有的类都是从对象类继承下来,字符串类当然也不例外,从父类继承下来就拥有了父类的一切属性与方法,所以斯汀类里面也有等于()方法,并且还把这个继承下来的equals()方法方法重写了)下来的等于()方法,改变了这个方法默认的实现,
在字符串类里面是这样重写的equals()方法的实现的:用当前的这个字符串对象和指定的字符串对象比较,指定的字符串对象不能为空并且这个对象的字符序列和当前这个字符串对象的字符串序列一样,如果这些条件都满足,那么这两个字符串对象就是相等的。
因此这里的S2已经满足了条件,所以打印出来的结果是正确的。
以后在某一个类里面比较两个对象是否相等时,首先去API文档里面查找这个类是否重写了从对象类继承下来的的equals()方法。如果重写了的equals()方法,那么在比较两个对象是否相等时调用的就是重写以后的equals()方法方法,如果没有重写,那么调用时就是直接调用从对象类里面的继承下来的那个的equals()方法,并且采用的equals()方法默认的实现去比较两个对象是否相等。因此每一个类都可以根据需要对从对象类继承下来的的equals()方法进行重写。
对于在API文档里面找某个类,如果一个类不用引入包就可以直接使用,那么这个类肯定是在java.lang中这个包里面,如这里的字符串类,直接就可以使用了,所以字符串类一定是在java.lang中这个包里面。使用某个类时看这个类引入的是哪个包,然后就去这个包里面找这个类,不用引入包的类一定是位于java.lang中里面,直接去的java。郎里面找就可以了。
总结:比较两个对象是否相等,我们采用的equals()方法,判断两个对象是否相等的条件是由我们重写的equals()方法的实现后定义的,这样就可以比较灵活地使用的equals()方法在不同的类里面比较位于同一类下的两个对象是否相等了。
对java中的equals()方法的总结的更多相关文章
- 关于Java中的equals方法
关于Java中的equals方法 欢迎转载,但是请填写本人的博客园原址https://www.cnblogs.com/JNovice/p/9347099.html 一.什么是equals方法 equa ...
- 【转】彻底弄懂Java中的equals()方法以及与"=="的区别
彻底弄懂Java中的equals()方法以及与"=="的区别 一.问题描述:今天在用Java实现需求的时候,发现equals()和“==”的功能傻傻分不清,导致结果产生巨大的偏差. ...
- 一文搞懂--Java中重写equals方法为什么要重写hashcode方法?
Java中重写equals方法为什么要重写hashcode方法? 直接看下面的例子: 首先我们只重写equals()方法 public class Test { public static void ...
- Java中“==”与equals方法的区别
1. 用“==”比较两个变量,如果两个变量是基本类型变量,且都是数值类,则值相等就返回true 如果两个变量是引用型变量,则两个对象的地址一样,即指向同一个对象,则返回true 2.equals:St ...
- JAVA中复写equals方法
在JAVA中“==”用于比较两个引用对象的地址是否相同.但是如果我们想比较两个对象的内容是否相同,通常会覆写equals方法.equals方法用来比较两个对象的内容是否相等. package org. ...
- JAVA中“==”与equals()方法区别
equals 方法是 java.lang.Object 类的方法 有两种用法说明: ()对于字符串变量来说,使用"=="和"equals()"方法比较字符串时, ...
- 如何在Java中避免equals方法的隐藏陷阱
摘要 本文描述重载equals方法的技术,这种技术即使是具现类的子类增加了字段也能保证equal语义的正确性. 在<Effective Java>的第8项中,Josh Bloch描述了当继 ...
- 如何重写Java中的equals方法
Java中,只有8种基本类型不是对象,例如:4种整形类型(byte, short, int,long),2种浮点类型(flout, double),boolean, char不是对象,其他的所有类型, ...
- java中的equals方法
这个方法首先比较的是两个对象的地址是否相同,如果相同直接返回true, 否则, (1)如果是string类型的先比较是否是string类型,是的话,再比较是否长度相同,相同的话再比较,每个字符是否相同 ...
- Java中的equals方法和自定义比较器
Object中的equals()方法默认是按地址比较,而不按内容进行比较, public boolean equals(Object obj) { return (this == obj); } 在S ...
随机推荐
- Python数据分析(一): ipython 技巧!
不一定非得使用Jupyter Notebook,试试ipython命令行 安装 ipython 我只试过Windows 10环境下的. 1.安装python安装包之后,应该就有ipython了. 2. ...
- Docker系列教程01-Centos7安装新版Docker教程(10步)
最近一直忙于开发,没有时间好好总结一下docker的知识.其实现在docker的教程已经很多很多了,但是很多系统的教程都是基于Ubuntu系统,因为官方推荐使用Ubuntu系统啊,原因在于Ubuntu ...
- HTTP请求定义不同Content-Type及在SpringMVC如何接收(必看!!!)
前言最近在和三方对接的时候发现了一些问题,这也是我写这篇文章的原因.我大概花了三天时间把这些内容了解,实践,整理,然后分享给大家,希望对大家会有所帮助.废话不多说,在和三方对接的时候我们规定使用jso ...
- 「造个轮子」——cicada 设计一个配置模块
前言 在前两次的 cicada 版本中其实还不支持读取配置文件,比如对端口.路由的配置. 因此我按照自己的想法创建了一个 issue ,也收集到了一些很不错的建议. 最终其实还是按照我之前的想法来做了 ...
- JQ 放大镜
<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title> ...
- 并发编程(十四)—— ScheduledThreadPoolExecutor 实现原理与源码深度解析 之 DelayedWorkQueue
我们知道线程池运行时,会不断从任务队列中获取任务,然后执行任务.如果我们想实现延时或者定时执行任务,重要一点就是任务队列会根据任务延时时间的不同进行排序,延时时间越短地就排在队列的前面,先被获取执行. ...
- Python爬虫入门教程 27-100 微医挂号网专家团队数据抓取pyspider
1. 微医挂号网专家团队数据----写在前面 今天尝试使用一个新的爬虫库进行数据的爬取,这个库叫做pyspider,国人开发的,当然支持一下. github地址: https://github.com ...
- 在 Vue 结合 Axios 使用过程 中 post 方法,后台无法接受到数据问题
关于在 vue 中 使用 axios 相关 bug 首先,我们来看下 axios 的 github 传送门 axios 然后我们再介绍下 axios 的作者的 github 传送门 Matt 最后,我 ...
- 可视化面板LogDashboard使用log4net源
logdashboard现已支持log4net文件源,本示例源码在 https://github.com/liangshiw/LogDashboard/tree/master/samples/UseL ...
- centos6.5-vsftp搭建
我的机子是默认是没有的vsftp. yum install -y vsftp 创建账户专为ftp而生.useradd ftp01 更改账户不可登录系统.usermod -s /sbin/nologin ...