我们知道List是一种有序链表:List内部按照放入元素的先后顺序存放,并且每个元素都可以通过索引确定自己的位置。

List还提供了boolean contains(Object o)方法来判断List是否包含某个指定元素。此外,int indexOf(Object o)方法可以返回某个元素的索引,如果元素不存在,就返回-1。

我们来看一个例子:

import java.util.List;

----

public class Main {

public static void main(String[] args) {

List list = List.of("A", "B", "C");

System.out.println(list.contains("C")); // true

System.out.println(list.contains("X")); // false

System.out.println(list.indexOf("C")); // 2

System.out.println(list.indexOf("X")); // -1

}

}

这里我们注意一个问题,我们往List中添加的"C"和调用contains("C")传入的"C"是不是同一个实例?

如果这两个"C"不是同一个实例,这段代码是否还能得到正确的结果?我们可以改写一下代码测试一下:

import java.util.List;

----

public class Main {

public static void main(String[] args) {

List list = List.of("A", "B", "C");

System.out.println(list.contains(new String("C"))); // true or false?

System.out.println(list.indexOf(new String("C"))); // 2 or -1?

}

}

因为我们传入的是new String("C"),所以一定是不同的实例。结果仍然符合预期,这是为什么呢?

因为List内部并不是通过==判断两个元素是否相等,而是使用equals()方法判断两个元素是否相等,例如contains()方法可以实现如下:

public class ArrayList {

Object[] elementData;

public boolean contains(Object o) {

for (int i = 0; i < size; i++) {

if (o.equals(elementData[i])) {

return true;

}

}

return false;

}

}

因此,要正确使用List的contains()、indexOf()这些方法,放入的实例必须正确覆写equals()方法,否则,放进去的实例,查找不到。我们之所以能正常放入String、Integer这些对象,是因为Java标准库定义的这些类已经正确实现了equals()方法。

我们以Person对象为例,测试一下:

import java.util.List;

----

public class Main {

public static void main(String[] args) {

List list = List.of(

new Person("Xiao Ming"),

new Person("Xiao Hong"),

new Person("Bob")

);

System.out.println(list.contains(new Person("Bob"))); // false

}

}

class Person {

String name;

public Person(String name) {

this.name = name;

}

}

不出意外,虽然放入了new Person("Bob"),但是用另一个new Person("Bob")查询不到,原因就是Person类没有覆写equals()方法。

编写equals

如何正确编写equals()方法?equals()方法要求我们必须满足以下条件:

自反性(Reflexive):对于非null的x来说,x.equals(x)必须返回true;

对称性(Symmetric):对于非null的x和y来说,如果x.equals(y)为true,则y.equals(x)也必须为true;

传递性(Transitive):对于非null的x、y和z来说,如果x.equals(y)为true,y.equals(z)也为true,那么x.equals(z)也必须为true;

一致性(Consistent):对于非null的x和y来说,只要x和y状态不变,则x.equals(y)总是一致地返回true或者false;

对null的比较:即x.equals(null)永远返回false。

上述规则看上去似乎非常复杂,但其实代码实现equals()方法是很简单的,我们以Person类为例:

public class Person {

public String name;

public int age;

}

首先,我们要定义“相等”的逻辑含义。对于Person类,如果name相等,并且age相等,我们就认为两个Person实例相等。

因此,编写equals()方法如下:

public boolean equals(Object o) {

if (o instanceof Person) {

Person p = (Person) o;

return this.name.equals(p.name) && this.age == p.age;

}

return false;

}

对于引用字段比较,我们使用equals(),对于基本类型字段的比较,我们使用==。

如果this.name为null,那么equals()方法会报错,因此,需要继续改写如下:

public boolean equals(Object o) {

if (o instanceof Person) {

Person p = (Person) o;

boolean nameEquals = false;

if (this.name == null && p.name == null) {

nameEquals = true;

}

if (this.name != null) {

nameEquals = this.name.equals(p.name);

}

return nameEquals && this.age == p.age;

}

return false;

}

如果Person有好几个引用类型的字段,上面的写法就太复杂了。要简化引用类型的比较,我们使用Objects.equals()静态方法:

public boolean equals(Object o) {

if (o instanceof Person) {

Person p = (Person) o;

return Objects.equals(this.name, p.name) && this.age == p.age;

}

return false;

}

因此,我们总结一下equals()方法的正确编写方法:

先确定实例“相等”的逻辑,即哪些字段相等,就认为实例相等;

用instanceof判断传入的待比较的Object是不是当前类型,如果是,继续比较,否则,返回false;

对引用类型用Objects.equals()比较,对基本类型直接用==比较。

使用Objects.equals()比较两个引用类型是否相等的目的是省去了判断null的麻烦。两个引用类型都是null时它们也是相等的。

如果不调用List的contains()、indexOf()这些方法,那么放入的元素就不需要实现equals()方法。

练习

给Person类增加equals方法,使得调用indexOf()方法返回正常:

import java.util.List;

----

public class Main {

public static void main(String[] args) {

List list = List.of(

new Person("Xiao", "Ming", 18),

new Person("Xiao", "Hong", 25),

new Person("Bob", "Smith", 20)

);

boolean exist = list.contains(new Person("Bob", "Smith", 20));

System.out.println(exist ? "测试成功!" : "测试失败!");

}

}

class Person {

String firstName;

String lastName;

int age;

public Person(String firstName, String lastName, int age) {

this.firstName = firstName;

this.lastName = lastName;

this.age = age;

}

}

小结

在List中查找元素时,List的实现类通过元素的equals()方法比较两个元素是否相等,因此,放入的元素必须正确覆写equals()方法,Java标准库提供的String、Integer等已经覆写了equals()方法;

编写equals()方法可借助Objects.equals()判断。

如果不在List中查找元素,就不必覆写equals()方法。

python字符串equals方法_编写equals方法相关推荐

  1. python比较两个字符串相似度_详解Python 字符串相似性的几种度量方法

    字符串的相似性比较应用场合很多,像拼写纠错.文本去重.上下文相似性等. 评价字符串相似度最常见的办法就是:把一个字符串通过插入.删除或替换这样的编辑操作,变成另外一个字符串,所需要的最少编辑次数,这种 ...

  2. Java黑皮书课后题第7章:**7.34(对字符串中的字符排序)使用以下方法头编写一个方法,返回一个排序好的字符串。编写一个测试程序,提示用户输入一个字符串,显示排序好的字符串

    **7.34(对字符串中的字符排序)使用以下方法头编写一个方法,返回一个排序好的字符串.编写一个测试程序,提示用户输入一个字符串,显示排序好的字符串 题目 题目描述 破题 代码 运行实例 题目 题目描 ...

  3. Java黑皮书课后题第6章:*6.23(指定字符的出现次数)使用下面的方法头编写一个方法,找到一个字符串中指定字符的出现次数。编写一个测试程序,提示用户输入一个字符串以及一个字符,显示该字符出现次数

    6.23(指定字符的出现次数)使用下面的方法头编写一个方法,找到一个字符串中指定字符的出现次数.编写一个测试程序,提示用户输入一个字符串以及一个字符,显示该字符出现次数 题目 题目描述 破题 代码 运 ...

  4. Java黑皮书课后题第8章:*8.2(求矩阵对角线元素的和)使用下面的方法头编写一个方法,求n*n的double类型矩阵中主对角线上所有数字的和。编写一个程序,读取一个4*4的矩阵,显示主对角线和

    *8.2(求矩阵对角线元素的和)使用下面的方法头编写一个方法,求n*n的double类型矩阵中主对角线上所有数字的和.编写一个程序,读取一个4*4的矩阵,显示主对角线和 题目 题目描述与运行示例 破题 ...

  5. Java黑皮书课后题第8章:*8.1(求矩阵中各列数字的和)使用下面的方法头编写一个方法,求矩阵中特定列的所有元素的和。编写一个测试程序,读取3*4矩阵,显示每列元素的和

    *8.1(求矩阵中各列数字的和)使用下面的方法头编写一个方法,求矩阵中特定列的所有元素的和.编写一个测试程序,读取3*4矩阵,显示每列元素的和 题目 题目描述与运行示例 破题 代码 题目 题目描述与运 ...

  6. Java黑皮书课后题第7章:7.10(找出最小元素的下标)使用下面的方法头编写一个方法,求出一个整数数组中的最小元素下标。编写测试程序,提示用户输入10个数字,调用这个方法返回最小值的下标(多个则最小

    7.10(找出最小元素的下标)使用下面的方法头编写一个方法,求出一个整数数组中的最小元素下标.编写测试程序,提示用户输入10个数字,调用这个方法返回最小值的下标(多个则返回最小的下标) 题目 题目描述 ...

  7. Java黑皮书课后题第7章:7.9(找出最小元素)使用下面的方法头编写一个方法,求出一个整数数组中的最小元素。编写测试程序,提示用户输入10个数字,调用这个方法返回最小值,并显示这个最小值

    7.9(找出最小元素)使用下面的方法头编写一个方法,求出一个整数数组中的最小元素.编写测试程序,提示用户输入10个数字,调用这个方法返回最小值,并显示这个最小值 题目 题目描述与运行示例 破题 代码 ...

  8. python字符串拼接数字_解决Python中字符串和数字拼接报错的方法

    解决Python中字符串和数字拼接报错的方法 前言 众所周知Python不像JS或者PHP这种弱类型语言里在字符串连接时会自动转换类型,如果直接将字符串和数字拼接会直接报错. 如以下的代码: # co ...

  9. [转载] Python字符串isdecimal() isdigit()isnumeric()等判断方法的区分。

    参考链接: Python 字符串String | isidentifier 在Python中,字符串的操作是非常基础而且用途也极为广泛,学会字符串的操作,会让你节省许多不必花费的时间,下面简单介绍下P ...

最新文章

  1. 获取控制器 nextResponder的简单应用
  2. Fluent UDF【4】:C语言
  3. java ssh客户端_简单的Java SSH客户端
  4. 淮阴工学院计算机答辩,淮阴工学院关于做好2014级研究生学位论文答辩工作的通知...
  5. FTP协议的分析和扩展
  6. 190729知识笔记
  7. NYOJ ~ 58 ~ 最小步数(BFS)
  8. ElasticSearch2.1 基于空间位置geo_query距离计算
  9. Mysql中,int(10)和int(11)的区别
  10. elasticsearch的性能表现
  11. SDN技术的十个关键因素
  12. 【泛函】线性算子的谱论(数学知识用程序表达轻松学系列)
  13. 关于将图片转成base64的方法
  14. java 正则用法_Java正则用法
  15. android 5.1一键root工具箱,s大师一键root下载
  16. vue动态面包屑导航
  17. 对计算机的理解和认识
  18. C语言编程单片机相关资料,单片机的C语言编程资料.ppt
  19. Ubuntu中出现sudo:unknown uid 1000:who are you?
  20. 乌班图系统解决zip解压乱码问题

热门文章

  1. Nginx安装与启动
  2. 《青你2》 | 有了它,你pick的小姐姐不靠“奶”也能出道
  3. java wasm_Wasm介绍之2:指令集和栈
  4. 苹果CMS cpu跑满占用高开启Redis高速缓存加快访问速度性能调优【转】
  5. Word骨灰级操作大全,赶紧留一份。
  6. Qt Quick 3D系列(一):加载3d模型
  7. 第五人格pcmac_第五人格求生者人格天赋 人类人格天赋系统解析
  8. 借助Aspose.BarCode,轻松实现QR和众多二维码在线扫描
  9. shell实现替换空格
  10. VMware虚拟机 Centos7 配置静态IP和DNS