这周我使用list.contains,发现new了相同数据的实体类,使用contains会判断为不相同。
原因是,实体类的equals的比较与我们期望的不一样,此时需要重写equals。
以下面这个定义为例,我们有一个实体类Info。
public class Info {
// 姓名
private String name;
// 年龄
private String age;
// 性别
private String sex;
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public String getAge() { return age; }
public void setAge(String age) { this.age = age; }
public String getSex() { return sex; }
public void setSex(String sex) { this.sex = sex; }
}
public static void main(String[] args) {
Info info = new Info();
info.setName("小张");
List<Info> list = new ArrayList<Info>();
list.add(info);
Info info2 = new Info();
info2.setName("小张");
if (list.contains(info2)) {
System.out.println("Exist.");
} else {
System.out.println("Non-exist.");
}
}
运行结果:
Non-exist.
我们看一下Info默认的equals:
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Info info = (Info) o;
if (!name.equals(info.name)) return false;
if (!age.equals(info.age)) return false;
return sex.equals(info.sex);
}
以name.equals(info.name)为例,这里的equals调用的是String的public boolean equals(Object anObject)。
因为age、sex没有设置,所以这里会抛出空指针的异常,导致结果返回false。
所以我重写equals —> 当name、info.name不全为空,且name ≠ info.name 时,返回false:
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Info info = (Info) o;
if (!(null == name && null == info.name) && !name.equals(info.name)) return false;
if (!(null == age && null == info.age) && !age.equals(info.age)) return false;
if (null == sex && null == info.sex) return true;
return sex.equals(info.sex);
}