平等还是认同?

将对象存储在集合中时,同一对象永远不能添加两次非常重要。 这是集合的核心定义。 在Java中,使用两种方法来确定两个引用的对象是否相同或它们是否可以同时存在于同一Set中。 equals()和hashCode()。 在本文中,我将解释平等与同一性之间的区别,并探讨它们在彼此之间所具有的一些优势。

Java提供了这两种方法的标准实现。 标准的equals()方法被定义为“身份”比较方法。 这意味着它将比较两个内存引用以确定它们是否相同。 因此,存储在内存中不同位置的两个相同的对象将被视为不相等。 如果使用Object类的源代码,可以使用==-operator进行比较。

public boolean equals(Object obj) {return (this == obj);
}

hashCode()方法由虚拟机作为本机操作实现,因此在代码中不可见,但通常将其实现为简单地返回内存引用(在32位体系结构上)或以32位模表示内存引用(在64位体系结构上)。

许多程序员在设计类时选择要做的一件事就是用不同的相等性定义覆盖此方法,在该方法中,您不查看内存引用,而是查看两个实例的值以查看它们是否可以相等。 这是一个例子:

import java.util.Objects;
import static java.util.Objects.requireNonNull;public final class Person {private final String firstname;private final String lastname;public Person(String firstname, String lastname) {this.firstname = requireNonNull(firstname);this.lastname  = requireNonNull(lastname);}@Overridepublic int hashCode() {int hash = 7;hash = 83 * hash + Objects.hashCode(this.firstname);hash = 83 * hash + Objects.hashCode(this.lastname);return hash;}@Overridepublic boolean equals(Object obj) {if (this == obj) return true;if (obj == null) return false;if (getClass() != obj.getClass()) return false;final Person other = (Person) obj;if (!Objects.equals(this.firstname, other.firstname)) {return false;} else return Objects.equals(this.lastname, other.lastname);}
}

这种比较称为“平等”(与之前的“身份”相比)。 只要两个人的名字和姓氏相同,就将被视为相等。 例如,这可以用于从输入流中筛选出重复项。 请记住,如果您覆盖equals()方法,则还应该始终覆盖hashCode()方法!

平等

现在,如果您选择平等而不是身份,那么您将需要考虑一些事项。 您必须问自己的第一件事是:具有相同属性的此类的两个实例是否一定相同? 对于上述人员,我会拒绝。 有一天,您的系统中很有可能会有两个姓氏和名字相同的人。 即使您继续添加更多的个人信息,例如生日或喜欢的颜色,您迟早也会发生冲突。 另一方面,如果您的系统正在处理汽车,并且每辆汽车都包含对“模型”的引用,则可以安全地假设,如果两辆汽车都是黑色Tesla模型S,则即使对象是相同的,它们也可能是同一模型。存储在内存中的不同位置。 这是平等可能很好的一个例子。

import java.util.Objects;
import static java.util.Objects.requireNonNull;public final class Car {public static final class Model {private final String name;private final String version;public Model(String name, String version) {this.name    = requireNonNull(name);this.version = requireNonNull(version);}@Overridepublic int hashCode() {int hash = 5;hash = 23 * hash + Objects.hashCode(this.name);hash = 23 * hash + Objects.hashCode(this.version);return hash;}@Overridepublic boolean equals(Object obj) {if (this == obj) return true;if (obj == null) return false;if (getClass() != obj.getClass()) return false;final Model other = (Model) obj;if (!Objects.equals(this.name, other.name)) {return false;} else return Objects.equals(this.version, other.version);}}private final String color;private final Model model;public Car(String color, Model model) {this.color = requireNonNull(color);this.model = requireNonNull(model);}public Model getModel() {return model;}
}

如果两辆汽车的内存地址相同,则它们被认为是相同的。 另一方面,只要它们具有相同的名称和版本,则认为它们的模型相同。 这是一个例子:

final Car a = new Car("black", new Car.Model("Tesla", "Model S"));
final Car b = new Car("black", new Car.Model("Tesla", "Model S"));System.out.println("Is a and b the same car? " + a.equals(b));
System.out.println("Is a and b the same model? " + a.getModel().equals(b.getModel()));// Prints the following:
// Is a and b the same car? false
// Is a and b the same model? true

身分识别

选择平等而不是身份的一种风险是,它可能会导致分配比堆上更多的对象。 只需看上面的汽车示例。 对于我们创建的每辆汽车,我们还为模型分配内存中的空间。 即使Java通常优化了字符串分配以防止重复,对于始终相同的对象仍然是一定的浪费。 将内部对象转换为可以使用身份比较方法进行比较并且同时避免不必要的对象分配的一个简单技巧是将其替换为枚举:

public final class Car {public enum Model {TESLA_MODEL_S ("Tesla", "Model S"),VOLVO_V70     ("Volvo", "V70");private final String name;private final String version;Model(String name, String version) {this.name    = name;this.version = version;}}private final String color;private final Model model;public Car(String color, Model model) {this.color = requireNonNull(color);this.model = requireNonNull(model);}public Model getModel() {return model;}
}

现在我们可以确定每种模型只会在内存中的某个位置存在,因此可以使用身份比较安全地进行比较。 然而,与此有关的一个问题是,这实际上限制了我们的可扩展性。 在此之前,可以在不修改Car.java文件中的源代码的情况下即时定义新模型的方法,但是现在我们已经将自己锁定在一个枚举中,该枚举通常应该保持不变。 如果需要这些属性,那么对等比较可能更适合您。

最后一点,如果您重写了类的equals()和hashCode()方法,后来又想根据身份将其存储在Map中,则始终可以使用IdentityHashMap结构。 即使equals()和hashCode()方法已被覆盖,它也将使用内存地址来引用其键。

翻译自: https://www.javacodegeeks.com/2016/03/equality-vs-identity.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/354631.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

二次优化大招(由泰勒公式推出最值条件)

经过前两篇的铺垫,这一篇迎来了高潮。先说一句:特征值大法好! 在开始正文之前,先看以下简单的推导 有了这些,理解下面的泰勒公式推出最值条件就容易了 转载于:https://www.cnblogs.com/Mr-ZeroW/p/7764916.html

前端自动化测试浅析

前言:测试简介前端常见的问题:修改某个模块功能时,其它模块也受影响,很难快速定位bug多人开发代码越来越难以维护不方便迭代,代码无法重构代码质量差增加自动化测试后:我们为核心功能编写测试后可以保障项目…

使用SpringData出现java.lang.AbstractMethodError

最近学习一下SpringData,在添加SpringData支持的时候,出现了这样的问题: SpringData需要的jar有:spring-data-jpa.jar spring-data-commons.jar slf4j-api.jar 没有添加slf4j也会出现一个异常,不过那个异常说的非常明确&#xf…

2021这份电子书单请收好(品类齐全)!

2021年已经快过去一个月了,今年的读书计划有没有安排上,这里有份长长的书单,多年的积累,品类齐全,赶快从中挑几本加入今年的读书计划里。•《货币战争2:金权天下》 - 宋鸿兵.mobi•《圣经》中英对照豪华版 …

在Kotlin中使用libGDX

最近,我一直在阅读有关不同语言的信息,以及它们可以为已经拥挤的软件开发人员带来什么,而一种语言对我来说很突出:Kotlin。 ( https://kotlinlang.org/ ) 这是一种相对较新的语言(成立于2011年…

练习1-2:编写一个 JAVA 程序,实现输出考试成绩的前三名。

1、 考试成绩已保存在数组 scores中,数组元素依次为 89 , -23 , 64 , 91 , 119 , 52 , 732、 要求通过自定义方法来实现成绩排名并输出操作,将成绩数组作为参数传入3、 要求判断成绩的有效性( 0—100 ),如果成绩无效&a…

【重学JS系列】slice用法大合集

让我们回顾下slice的日常用法slice 工作原理在深入研究一些更高级的用法之前,让我们看一下slice方法的基础知识。如MDN文档,slice 是数组上的一个方法,它最多有两个参数:arr.slice([begin[, end]])begin从该索引处开始提取原数组中的元素,如果…

redux-4-ways

https://medium.com/react-native-training/redux-4-ways-95a130da0cdc转载于:https://www.cnblogs.com/skating/p/7774090.html

【Github开源】一站搞定各种开发文档

开发者的苦恼:经常要在多个API文档中切换,浏览器书签栏收藏各种语言相关的接口说明文档。无意中在Github上发现DevDocs[1]这个开源项目,它是一个把所有开发相关的文档以web的形式做了一个综合的网站,并提供搜索,离线访…

javafx 表单_JavaFX 2:创建登录表单

javafx 表单在本教程中,我将使用JavaFX 2和CSS设计一个外观漂亮的Login Form 。 它是经典的登录表单,带有用户名和密码以及登录按钮。 为了遵循本教程,我强烈建议您查看以下这些教程: Eclipse IDE中的JavaFX 2入门 JavaFX 2&…

中高级JavaScript易错面试题

写出下题的输出 1、函数的实参与形参length var length 10; function fn() {console.log(this.length); } var obj {length: 5,method: function(fn) {fn();arguments[0]();} }; console.log(obj.method(fn, 1));  // 0 2 我们都知道,[1, 2, 3].length可以得到3…

静音设计模式

您最近是否遵循Mute-Design-Pattern™编写了大量代码? 例如 try {complex();logic();here(); } catch (Exception ignore) {// Will never happen heheSystem.exit(-1); }Java 8有一个更简单的方法! 只需将这个非常有用的工具添加到您的Utilities或Hel…

【详细教程】教你如何使用Node + Express + Typescript开发一个应用

Express是nodejs开发中普遍使用的一个框架,下面要谈的是如何结合Typescript去使用。 目标 我们的目标是能够使用Typescript快速开发我们的应用程序,而最终我们的应用程序却是编译为原始的JavaScript代码,以由nodejs运行时来执行。 初始化设置…

结构型模式 适配器模式

结构型模式 适配器模式 适用于: 是将一个类的接口转换成客户希望的另外一个接口。使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。 /*** 结构型模式 适配器模式* Adapter模式也叫适配器模式,是构造型模式之一,通过Adapter模式可…

乐哥学AI_Python(二):Numpy索引,切片,常用函数

Numpy的索引和切片 ndarray对象的内容可以通过索引和切片查看和修改。 索引:ndarray对象中的元素索引基于0开始 切片:对数组里某个片段区域的描述 数组的切片也可以理解为原始数组的局部视图,都是指向内存中的原始数组,所以不同于…

仅使用HTML和CSS实现的标签云效果

标签云的效果在博客和网站上不难见到,它其实就是带有超链接的某些关键字,为了达到强调主题的作用。通常出现概率比较大或者受欢迎的标签文字显示比较大,相反的就显示的小。来源于TagCrowd.com我们就不去深入研究标签云带来的效率上的提升和可…

捍卫Java

因此,我们不时发布了一本电子书,名为“十大Java性能问题” 。 毫无例外,一些人回答说“问题是您正在使用Java”。 显然,Java一直在受到批评,人们已经预测了它的消亡已有一段时间了。 当然,它不像Python&am…

如何在客户端终止一个已经发出的HTTP请求

Javascript 异步编程得益于 Promise 的实现,它们极大地提高了Web开发的性能和体验。不过原生的 Promise 有个最大的缺点就是一旦请求发出去,我们就无法取消它。但是我们找到了另一种方法来实现。DOM 标准中添加了一个称为 AbortController 的新的控制器&…

vuex的使用二

1.先看项目的目录结构 2.在main.js里需要引入store这个文件并挂在实例上 import store from ./store/store ............new Vue({el: #app,router,store,template: <App/>,components: { App } }) 3.store.js里引入action.js和mutation.js文件 // 状态管理器 import Vue…

收到短信验证码自动填充到表单,竟然是这么玩的

苹果系统上的App和网站可以实现来自短信的验证码自动填充表单的功能&#xff0c;通常你是怎么实现这个功能的&#xff1f;有一种实现方式可能你不知道&#xff0c;单纯的HTML标签就能实现&#xff0c;不需要任何的Javascript代码该特性第一次发布是在 WWDC 2018[1]&#xff1a;…