@EqualsAndHashCode

1、@Data注解包含了这些注解

* @see Getter
* @see Setter
* @see RequiredArgsConstructor
* @see ToString
* @see EqualsAndHashCode
* @see lombok.Value

1. 此注解会生成equals(Object other) 和 hashCode()方法。 
2. 它默认使用非静态,非瞬态的属性 
3. 可通过参数exclude排除一些属性 
4. 可通过参数of指定仅使用哪些属性 
5. 它默认仅使用该类中定义的属性且不调用父类的方法 
6. 可通过callSuper=true解决上一点问题。让其生成的方法中调用父类的方法。

另:@Data相当于@Getter @Setter @RequiredArgsConstructor @ToString @EqualsAndHashCode这5个注解的合集。

 

2、@Data就包含了EqualsAndHashCode 重写了hashcode和equals方法

3、EqualsAndHashCode  对于继承父类属性时需要注意的点

通过官方文档,可以得知,当使用@Data注解时,则有了@EqualsAndHashCode注解,那么就会在此类中存在equals(Object other) 和 hashCode()方法,且不会使用父类的属性,这就导致了可能的问题。 
比如,有多个类有相同的部分属性,把它们定义到父类中,恰好id(数据库主键)也在父类中,那么就会存在部分对象在比较时,它们并不相等,却因为lombok自动生成的equals(Object other) 和 hashCode()方法判定为相等,从而导致出错。

修复此问题的方法很简单: 
1. 使用@Getter @Setter @ToString代替@Data并且自定义equals(Object other) 和 hashCode()方法,比如有些类只需要判断主键id是否相等即足矣。 
2. 或者使用在使用@Data时同时加上@EqualsAndHashCode(callSuper=true)注解。

 

4、测试

@Data
//@EqualsAndHashCode(callSuper=true)
public class UcsAccountLog extends  BaseVO implements Serializable {
 
    private static final long serialVersionUID = 1L;
 
    private String id;
    
}

 

@Data
public class BaseVO {
    public String version;
}

 

/**
     * 首先,是为了支持哈希表类的如之类的底层使用了哈希表的类。
     * Java Object规范中int hashCode()方法的约定类容有三个:
     * <p>
     * (1)              只要对象的equals方法所用到的信息没有修改,那么hashCode方法必须始终如一的返回一个同一整数,在同一应用程序中多次执行中,每一次执行可以不一样。
     * <p>
     * (2)              如果两个对象的equals方法想到,那么每一个对象单独调用hashCode返回的数字必须相等。
     * <p>
     * (3)              如果两个对象的equals方法不相等,hashCode方法的返回值可以相等,给不同的对象返回不同的值可以提高哈希表的性能。
     * 所有重写 equals的同时 必须重写hashcode
     * @param args
     */
    public static void main(String[] args) {
        UcsAccountLog ucsAccountLog = new UcsAccountLog();
        ucsAccountLog.setVersion("1");
        ucsAccountLog.setId("2");
        UcsAccountLog ucsAccountLog2 = new UcsAccountLog();
        ucsAccountLog2.setVersion("2");
        ucsAccountLog2.setId("2");
        HashSet set = new HashSet();
        set.add(ucsAccountLog);
        set.add(ucsAccountLog2);
        System.out.println(set.size());//1
        System.out.println(ucsAccountLog.hashCode() == ucsAccountLog2.hashCode());//true  因为重写了hashcode 所有这里hashcode是相等的
        //如果没有重写 hashcode是根据内存地址值
        System.out.println(ucsAccountLog.equals(ucsAccountLog2));//true
        //根据hash
        System.out.println(ucsAccountLog == ucsAccountLog2);//false  ==比较的是对象的地址
 
    }

 如果子类 

//@EqualsAndHashCode(callSuper=true)

没有添加这一行,则输出结果是 true true false 和我们期望的结果不符合 

 因为@EqualsAndHashCode默认不继承父类 就是说 重写hashcode和equals的时候 不包含父类的字段值,所有只会比较自己对象里面的字段值  显然这是错误的

打开注解 则输出正确结果 false false false

 

 

 

posted @ 2019-06-16 15:09  Andrew_F  阅读(12963)  评论(1编辑  收藏  举报