现在有两个对象,他们的一个属性是list,很明显两个对象的list里面的对象,都是相等的,只是这2个list里面的顺序不一致,导致这2个对象被判断为不相等啦,这就是问题,现在要解决这个问题。

问题图如下:

大师兄

可以看到这2个对象的呢个list属性里面数据,咱看起来是一样的,但是经过equals之后,返回的是false。

所以,需要自己重写equals方法和hashcode方法,这2个方法一般是一起重写的。

然后,问题的关键就在于,怎么判断2个list集合是否相等。不考虑顺序。

看代码:

先是这2个对象的代码

TableMetaData

package com.lxk.model; 
 
 
import lombok.AllArgsConstructor; 
import lombok.Data; 
import lombok.NoArgsConstructor; 
 
import java.util.Collection; 
import java.util.Objects; 
 
/** 
 * Table metadata. 
 * 
 * @author lxk 
 */ 
@Data 
@AllArgsConstructor 
@NoArgsConstructor 
public class TableMetaData { 
     
    private Collection<ColumnMetaData> columnMetaData; 
     
    @Override 
    public boolean equals(Object o) { 
        if (this == o) { 
            return true; 
        } 
        if (o == null || getClass() != o.getClass()) { 
            return false; 
        } 
        TableMetaData that = (TableMetaData) o; 
        if (columnMetaData.size() != that.getColumnMetaData().size()) { 
            return false; 
        } 
        return columnMetaData.containsAll(that.getColumnMetaData()); 
    } 
 
    @Override 
    public int hashCode() { 
 
        return Objects.hash(columnMetaData); 
    } 
} 

然后是,list里面装的对象的代码

 
package com.lxk.model; 
 
 
import lombok.AllArgsConstructor; 
import lombok.Data; 
import lombok.NoArgsConstructor; 
 
import java.util.Objects; 
 
/** 
 * Column metadata. 
 * 
 * @author lxk 
 */ 
@Data 
@NoArgsConstructor 
@AllArgsConstructor 
public final class ColumnMetaData { 
     
    private String columnName; 
     
    private String columnType; 
     
    private boolean primaryKey; 
 
    @Override 
    public boolean equals(Object o) { 
        if (this == o) { 
            return true; 
        } 
        if (o == null || getClass() != o.getClass()) { 
            return false; 
        } 
        ColumnMetaData that = (ColumnMetaData) o; 
        return primaryKey == that.primaryKey && 
                Objects.equals(columnName, that.columnName) && 
                Objects.equals(columnType, that.columnType); 
    } 
 
    @Override 
    public int hashCode() { 
 
        return Objects.hash(columnName, columnType, primaryKey); 
    } 
 
    @Override 
    public String toString() { 
        return "ColumnMetaData{" + 
                "columnName='" + columnName + '\'' + 
                ", columnType='" + columnType + '\'' + 
                ", primaryKey=" + primaryKey + 
                '}'; 
    } 
} 

最后就是main方法啦

package com.lxk.commonTest; 
 
import com.google.common.collect.Lists; 
import com.lxk.model.ColumnMetaData; 
import com.lxk.model.TableMetaData; 
 
import java.util.Collection; 
 
/** 
 * 测试equals的各种情况 
 * 
 * @author lxk on 2018/9/17 
 */ 
public class EqualsTest { 
    public static void main(String[] args) { 
        testCollectionEqual(); 
    } 
 
    /** 
     * 比较2个集合是否是相同的,不考虑集合的顺序,只要里面的元素确实相同即可。 
     */ 
    private static void testCollectionEqual() { 
        ColumnMetaData c1 = new ColumnMetaData("age1", "name", true); 
        ColumnMetaData c2 = new ColumnMetaData("age2", "name", true); 
        ColumnMetaData c3 = new ColumnMetaData("age3", "name", true); 
        ColumnMetaData c4 = new ColumnMetaData("age4", "name", true); 
        Collection<ColumnMetaData> list1 = Lists.newArrayList(c1, c2, c3); 
        Collection<ColumnMetaData> list2 = Lists.newArrayList(c2, c3, c1); 
        TableMetaData table1 = new TableMetaData(list1); 
        TableMetaData table2 = new TableMetaData(list2); 
        boolean equals = table1.equals(table2); 
        System.out.println(equals); 
    } 
 
} 

如上操作之后,这个判断就可以得到2个对象,是真的相等啦,不会因为list里面值的顺序不一致,而返回false啦。

大师兄

这个问题简单吗,简单就简单吧,我就喜欢简单的问题。老铁们别喷。

至于,重写hashcode和equals方法,这里面学问大啦。

有兴趣的可以参考如下的几个链接,涨涨姿势:

Java:重写equals()和hashCode()

Java Objects.equals()的作用---避免空指针异常

JAVA中重写equals()方法的同时要重写hashcode()方法

java中的==、equals()、hashCode()源码分析

为什么HashSet不能重复以及具体原理源码分析

建立一个对象模型的时候,为什么要重写hashCode与equals方法

Java 1.8 HashMap 源码中 put()方法详解

上面这几个都是我当时总结的,有的是转载的,有的是自己亲自实践的。有耐心的,不清楚的,可以了解一下。


评论关闭
IT序号网

微信公众号号:IT虾米 (左侧二维码扫一扫)欢迎添加!