我们知道List是一种有序链表:List内部按照放入元素的先后顺序存放,并且每个元素都可以通过索引确定自己的位置。
List还提供了boolean contains(Object o)方法来判断List是否包含某个指定元素。<br />此外,int indexOf(Object o)方法可以返回某个元素的索引,如果不存在此元素,则返回-1`

  1. public class Main{
  2. public static void main(String[] args){
  3. List<String> list = List.of("A","B","C");
  4. System.out.println(list.contains("C"));
  5. System.out.println(list.indexOf("C"));//2
  6. }
  7. }

这里我们注意到一个问题,我们往List中添加的"C"和调用contains("C")传入的"C"是不是同一个实例?
如果这两个"C"不是同一个实例,这段代码是否还能得到正确的结果?

  1. List<String> list = List.of("A","B","C");
  2. System.out.println(list.contains(new String("C"));
  3. System.out.println(list.indexOf(new String("C"));

通过运行可知,返回true。List内部并不是通过==来判断两个元素是否相等,而是使用equals()方法判断两个元素是否相等,例如contains()方法可以实现如下

  1. public class ArrayList{
  2. Object[] elementData;
  3. public boolean contains(Object o) {
  4. for(int i =0; i< elementData.length;i++){
  5. if(o.equals(elementData[i])) {
  6. return true;
  7. }
  8. }
  9. return false;
  10. }

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

Person对象为例

  1. public class Main{
  2. public static void main(String[] args){
  3. List<Person> list = list.of(new Person("xiaoming"),new Person("xiaohong"),new Person("Bob"));
  4. System.out.println(list.contains(new Person("Bob"));
  5. }
  6. }
  7. class Person {
  8. String name;
  9. public Person(String name){this.name= name;}
  10. }

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

编写equals

equals()方法要求我们必须满足以下条件

  • 自反性(Reflexive):对于非nullx来说,x.equals(x)必须返回true
  • 对称性(Symmetric):对于非nullxy来说,如果,x.equals(y)true,那么y.equals(x)也必须为true
  • 传递性(Transitive): 对于非nullx、y、z来说,如果x.equals(y) = true,x.equals(z) = true那么y.equlas(z)也必须为true
  • 一致性(Consistent):对于非nullx 、y来说,只要xy的状态不变,则x.equals(y)总是一致地返回true或者false
  • null的比较,x.equals(null)永远返回false

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

  1. public class Person {
  2. public String name;
  3. public int age;
  4. }

首先,我们要定义“相等”的逻辑含义。
对于Person类,如果name相等,并且age相等,那么我们就认为两个Person实例相等。
因此,编写equals()方法

  1. public boolean equals(Object o) {
  2. if(o instanceof Person){
  3. Person p =(Person) o;
  4. return this.name.equals(p.name) && this.age == p.age;
  5. }
  6. return false;
  7. }

对于引用字段比较,我们使用equals(),对于基本类型使用==
如果this.namenull,那么equals()方法会报错,因此,需要继续改写

  1. public boolean equals(Object o) {
  2. if (o instanceof Person){
  3. Person p = (Person) o;
  4. boolean nameEquals = false;
  5. if(this.name == null && p.name == null){
  6. nameEquals = true;
  7. }
  8. if(this.name != null){
  9. nameEquals = this.name.equals(p.name);
  10. }
  11. return nameEquals && this.age == p.age;
  12. }
  13. return false;
  14. }

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

  1. public boolean equals(Object o) {
  2. if (o instanceof Person){
  3. Person p= (Person) o;
  4. return Objects.equals(this.name,p.name) && this.age == p.age;
  5. }
  6. return false;
  7. }

总结一下equals()方法的正确编写方法

  1. 先确定实例"相等"的逻辑,即哪些字段相等,就认为实例相等
  2. instanceof判断传入的待比较的Object是不是当前类型,如果是,继续比较,否则,返回false
  3. 对引用类型用Objects.equals()比较,对基本类型直接用==比较。

使用Objects.equals()比较两个引用类型是否相等的目的是活动了判断null的麻烦。两个引用类型都是null时它们也是相等的。
如果不调用Listcontains()indexOf()这些方法,那么放入元素就不需要 实现equals()方法。

练习
Person类增加equalas()方法,使得调用indexOf()方法返回正常

  1. public class Main {
  2. public static void main(String[] args){
  3. List<Person> list = list.of(
  4. new Person("xiao","ming",18);
  5. new Person("xiao","hong",25);
  6. new Person("xiao","jun",20);
  7. );
  8. boolean exist = list.contains(new Person("xiao","jun",20));
  9. System.out.println(exist ? "测试成功!" :"测试失败";
  10. }
  11. }
  12. @getter
  13. @setter
  14. class Person{
  15. private String firstName;
  16. private String lastName;
  17. private int age;
  18. public Person(String firstName, String lastName, int age){
  19. this.firstName = firstName;
  20. this.lastName = lastName;
  21. this.age = age;
  22. }
  23. @Override
  24. public boolean equals(Object o){
  25. if(o intanceof Person){
  26. Person p = (Person) o;
  27. return Objects.equals(this.firstName, p.getFirstName()) && Obejcts.equals(this.lastName, p.getLastName()) && this.age == p.getAge());
  28. }
  29. }
  30. }

小结

List中查找元素时,List的实现类通过元素的equals()方法比较两个元素是否相等,因此,放入的元素必须正确覆写equals()方法,Java标准库提供的StringInteger等已经覆写了equals()方法
编写equals()方法可以借助Objects.equals()判断
如果不在List中查找元素,就不必覆写equals()方法