Java各种比较对象的方式的对比总结


Posted in Java/Android onJune 20, 2021

一、==和!=操作符

让我们从==和!=开始可以分别判断两个Java对象是否相同的操作符。

1.1 原始类型(Primitives)

对于原始类型,相同意味着具有相等的值:

assertThat(1 == 1).isTrue();

感谢自动拆箱,在将原语值与其包装类型对应值进行比较时,也可以这样做:

Integer a = new Integer(1);
assertThat(1 == a).isTrue();

如果两个整数的值不同,==运算符将返回false,而!=运算符将返回true。

1.2 对象比较

假设我们要比较两个具有相同值的整数包装类型:

Integer a = new Integer(1);
Integer b = new Integer(1);

assertThat(a == b).isFalse();

通过比较两个对象,这些对象的值不是1,而是它们在堆栈中的内存地址,因为这两个对象都是使用new操作符创建的。如果我们把a分配给b,我们会得到不同的结果:

Integer a = new Integer(1);
Integer b = a;

assertThat(a == b).isTrue();

现在,让我们看看使用Integer#valueOf factory方法时会发生什么:

Integer a = Integer.valueOf(1);
Integer b = Integer.valueOf(1);

assertThat(a == b).isTrue();

在这种情况下,它们被认为是相同的。这是因为valueOf()方法将整数存储在缓存中,以避免创建太多具有相同值的包装器对象。因此,该方法为两个调用返回相同的整数实例。

对字符串也是一样:

assertThat("Hello!" == "Hello!").isTrue();

但是,如果它们是使用new操作符创建的,那么它们就不一样了。最后,两个空引用被认为是相同的,而任何非空对象将被认为与空对象不同:

assertThat(null == null).isTrue();
assertThat("Hello!" == null).isFalse();

当然,相等运算符的行为可能是有限的。如果我们想比较两个映射到不同地址的对象,并根据它们的内部状态将它们视为相等,那该怎么办?我们将在下一节中看到如何操作。

二、Object的equals方法

现在,让我们用equals()方法讨论一个更广泛的相等概念。这个方法是在Object类中定义的,以便每个Java对象都继承它。默认情况下,它的实现比较对象内存地址,因此它的工作方式与==运算符相同。但是,我们可以重写这个方法来定义相等对我们的对象意味着什么。

首先,让我们看看它对现有对象(如Integer)的表现:

Integer a = new Integer(1);
Integer b = new Integer(1);

assertThat(a.equals(b)).isTrue();

当两个对象相同时,该方法仍然返回true。我们应该注意,我们可以传递一个空对象作为方法的参数,但是当然,不能作为调用方法的对象。我们可以对自己的对象使用equals()方法。假设我们有一个Person类:

public class Person {
    private String firstName;
    private String lastName;

    public Person(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }
}

我们可以重写该类的equals()方法,以便根据两个人的内部详细信息进行比较:

@Override
public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;
    Person that = (Person) o;
    return firstName.equals(that.firstName) &&
      lastName.equals(that.lastName);
}

三、Objects的静态方法equals

现在让我们看看Objects的equals静态方法。我们前面提到过,不能使用null作为第一个对象的值,否则会抛出NullPointerException。Objects助手类的equals()方法解决了这个问题。它接受两个参数并比较它们,同时处理空值。

让我们再次比较Person对象:

Person joe = new Person("Joe", "Portman");
Person joeAgain = new Person("Joe", "Portman");
Person natalie = new Person("Natalie", "Portman");

assertThat(Objects.equals(joe, joeAgain)).isTrue();
assertThat(Objects.equals(joe, natalie)).isFalse();

如前所述,该方法处理空值。因此,如果两个参数都为null,则返回true;如果只有一个参数为null,则返回false。这真的很方便。假设我们想给Person类添加一个可选的出生日期:

public Person(String firstName, String lastName, LocalDate birthDate) {
    this(firstName, lastName);
    this.birthDate = birthDate;
}

然后,我们必须更新equals()方法,但是要处理Null。我们可以将此条件添加到equals()方法中:

birthDate == null ? that.birthDate == null : birthDate.equals(that.birthDate);

但是,如果我们在类中添加许多可为null的字段,它可能会变得非常混乱。在equals()实现中使用Objects#equals方法更加简洁,并且提高了可读性:

Objects.equals(birthDate, that.birthDate);

四、Comparable接口

比较逻辑也可用于给对象排序。可比较接口允许我们通过确定一个对象是大于、等于还是小于另一个对象来定义对象之间的顺序。

Compariable接口是泛型的,只有一个方法compareTo(),该方法接受泛型类型的参数并返回int。如果当前值小于参数,则返回负值;如果它们相等,则返回0;否则返回正值。

比方说,在我们的Person类中,我们希望按Person对象的姓氏进行比较:

public class Person implements Comparable<Person> {
    //...

    @Override
    public int compareTo(Person o) {
        return this.lastName.compareTo(o.lastName);
    }
}

如果使用姓氏大于此的人调用compareTo()方法,则返回负int;如果姓氏相同,则返回零;否则返回正int。

五、Comparator接口

Comparator接口是泛型的,并且有一个compare方法,该方法接受该泛型类型的两个参数并返回一个整数。我们在前面的可比较接口中已经看到了这种模式。

比较器相似;但是,它与类的定义是分离的。因此,我们可以为一个类定义任意多个比较器,其中我们只能提供一个可比较的实现。

假设我们有一个在网页中有一个展示人信息的表格,我们想让用户能够按名字而不是姓氏对他们进行排序。如果我们还想保持当前的实现,那么使用Comparable是不可能的,但是我们可以实现自己的比较器。

让我们创建一个Person Comparator,它将只根据他们的名字进行比较:

Comparator<Person> compareByFirstNames = Comparator.comparing(Person::getFirstName);

现在让我们用这个比较器对一组人进行排序:

Person joe = new Person("Joe", "Portman");
Person allan = new Person("Allan", "Dale");

List<Person> people = new ArrayList<>();
people.add(joe);
people.add(allan);

people.sort(compareByFirstNames);

assertThat(people).containsExactly(allan, joe);

在compareTo()实现中,可以使用Comparator接口上的其他方法:

@Override
public int compareTo(Person o) {
    return Comparator.comparing(Person::getLastName)
      .thenComparing(Person::getFirstName)
      .thenComparing(Person::getBirthDate, Comparator.nullsLast(Comparator.naturalOrder()))
      .compare(this, o);
}

在这种情况下,我们首先比较姓,然后比较名。然后,我们比较出生日期,但是由于它们是可空的,我们必须说明如何处理它,所以我们给出了第二个参数,告诉它们应该根据它们的自然顺序进行比较,但是空值最后才是。

六、使用Apache Commons

现在让我们看看apachecommons库。首先,让我们导入Maven依赖项:

<dependency>
    <groupId>org.apache.commons</groupId>
    <artifactId>commons-lang3</artifactId>
    <version>3.11</version>
</dependency>

6.1 ObjectUtils的notEqual方法

首先,我们来讨论ObjectUtils#notEqual方法。它需要两个对象参数,根据它们自己的equals()方法实现来确定它们是否相等。它还处理空值。

让我们重新使用我们的字符串示例:

String a = new String("Hello!");
String b = new String("Hello World!");

assertThat(ObjectUtils.notEqual(a, b)).isTrue();

应该注意,ObjectUtils有一个equals()方法。但是,自从Java7出现Objects#equals之后,这种方法就被弃用了。

6.2 ObjectUtils的compare方法

现在,让我们用ObjectUtils#compare方法比较对象顺序。它是一个泛型方法,它接受该泛型类型的两个可比较参数并返回一个整数。

让我们再看看如何使用字符串:

String first = new String("Hello!");
String second = new String("How are you?");

assertThat(ObjectUtils.compare(first, second)).isNegative();

默认情况下,该方法通过将空值视为更大值来处理空值。它提供了一个重载版本,它提供了一个布尔参数来反转该行为,并将它们考虑得更小。

七、使用Guava

现在,我们来看看Guava。Guava 是 Google 的一个开源项目,包含许多 Google 核心 Java 常用库,如:集合 [collections] 、缓存 [caching] 、原生类型支持 [primitives support] 、并发库 [concurrency libraries] 、通用注解 [common annotations] 、字符串处理 [string processing] 与 I/O 等。

首先,让我们导入依赖项:

<dependency>
    <groupId>com.google.guava</groupId>
    <artifactId>guava</artifactId>
    <version>29.0-jre</version>
</dependency>

7.1 Objects的equal方法

与apache commons库类似,Google为我们提供了一种方法来确定两个对象是否相等,objects#equal。尽管它们有不同的实现,但它们返回相同的结果:

String a = new String("Hello!");
String b = new String("Hello!");

assertThat(Objects.equal(a, b)).isTrue();

尽管它没有被标记为deprecated,但是这个方法的JavaDoc说它应该被视为deprecated,因为java7提供了Objects#equals方法。

7.2 Comparison方法

现在,Guava库没有提供一个方法来比较两个对象(我们将在下一节中看到如何实现这一点),但它确实提供了比较原始值的方法。让我们看看Ints helper类的compare()方法是如何工作的:

assertThat(Ints.compare(1, 2)).isNegative();

通常,如果第一个参数小于、等于或大于第二个参数,则返回一个整数,该整数可能为负、零或正。除了bytes之外,所有基元类型都有类似的方法。

7.3 ComparisonChain类

最后,Guava库提供了ComparisonChain类,它允许我们通过一系列比较来比较两个对象。我们可以很容易地比较两个人的名字和姓氏:

Person natalie = new Person("Natalie", "Portman");
Person joe = new Person("Joe", "Portman");

int comparisonResult = ComparisonChain.start()
  .compare(natalie.getLastName(), joe.getLastName())
  .compare(natalie.getFirstName(), joe.getFirstName())
  .result();

assertThat(comparisonResult).isPositive();

底层比较是使用compareTo()方法实现的,因此传递给compare()方法的参数必须是原始类型或可比较的对象。

八、完整代码

Java对象比较

到此这篇关于Java各种比较对象的方式的对比总结的文章就介绍到这了,更多相关Java比较对象的方式内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
浅谈Java实现分布式事务的三种方案
Jun 11 Java/Android
详解JAVA中的OPTIONAL
Jun 14 Java/Android
Java Dubbo框架知识点梳理
Jun 26 Java/Android
Java基于字符界面的简易收银台
Jun 26 Java/Android
自从在 IDEA 中用了热部署神器 JRebel 之后,开发效率提升了 10(真棒)
Jun 26 Java/Android
Java比较两个对象中全部属性值是否相等的方法
Aug 07 Java/Android
关于maven依赖 ${xxx.version}报错问题
Jan 18 Java/Android
JVM的类加载器和双亲委派模式你了解吗
Mar 13 Java/Android
Elasticsearch Recovery 详细介绍
Apr 19 Java/Android
详解Android中的TimePickerView(时间选择器)的用法
Apr 30 Java/Android
详解Spring Security如何在权限中使用通配符
Jun 28 Java/Android
向Spring IOC 容器动态注册bean实现方式
Jul 15 Java/Android
Java Optional<Foo>转换成List<Bar>的实例方法
Jun 20 #Java/Android
详解Java实践之适配器模式
Java数据结构之链表相关知识总结
详解Java实践之抽象工厂模式
Java框架入门之简单介绍SpringBoot框架
详解Java实践之建造者模式
新手入门Jvm-- JVM对象创建与内存分配机制
You might like
在PHP中设置、使用、删除Cookie的解决方法
2013/05/06 PHP
php中使用getimagesize获取图片、flash等文件的尺寸信息实例
2014/04/29 PHP
解决CodeIgniter伪静态失效
2014/06/09 PHP
ThinkPHP之用户注册登录留言完整实例
2014/07/22 PHP
phpword插件导出word文件时中文乱码问题处理方案
2014/08/19 PHP
php实现无限级分类
2014/12/24 PHP
php动态添加url查询参数的方法
2015/04/14 PHP
PHP封装的PDO数据库操作类实例
2017/06/21 PHP
Javascript中String的常用方法实例分析
2015/06/13 Javascript
JavaScript中循环遍历Array与Map的方法小结
2016/03/12 Javascript
Bootstrap导航条可点击和鼠标悬停显示下拉菜单的实现代码
2016/06/23 Javascript
Webpack如何引入bootstrap的方法
2017/06/17 Javascript
详解vue模拟加载更多功能(数据追加)
2017/06/23 Javascript
动态Axios的配置步骤详解
2018/01/12 Javascript
详解ES6语法之可迭代协议和迭代器协议
2018/01/13 Javascript
JS实现读取xml内容并输出到div中的方法示例
2018/04/19 Javascript
LayUI表格批量删除方法
2018/08/15 Javascript
vue-cli 打包后提交到线上出现 &quot;Uncaught SyntaxError:Unexpected token&quot; 报错
2018/11/06 Javascript
Vue表单绑定的实例代码(单选按钮,选择框(单选时,多选时,用 v-for 渲染的动态选项)
2019/05/13 Javascript
实现高性能javascript的注意事项
2019/05/27 Javascript
js实现无缝轮播图特效
2020/05/09 Javascript
[01:41]DOTA2超级联赛专访YYF 称一辈子难忘TI2
2013/05/28 DOTA
[02:17]TI4西雅图DOTA2前线报道 啸天mik夫妻档解说
2014/07/08 DOTA
pycharm运行出现ImportError:No module named的解决方法
2018/10/13 Python
python3的数据类型及数据类型转换实例详解
2019/08/20 Python
Django数据结果集序列化并展示实现过程
2020/04/22 Python
Python Switch Case三种实现方法代码实例
2020/06/18 Python
浅谈Python __init__.py的作用
2020/10/28 Python
HTML5移动端开发遇见的东西
2019/10/11 HTML / CSS
html5 初试 indexedDB(推荐)
2016/07/21 HTML / CSS
简述数组与指针的区别
2014/01/02 面试题
大学生就业自荐信
2013/10/26 职场文书
在校生自我鉴定
2014/01/23 职场文书
总经理岗位职责说明书
2014/07/30 职场文书
政协调研汇报材料
2014/08/15 职场文书
亲戚关系证明
2015/06/24 职场文书