我有两个用户定义的对象,它们的内容相同。但是assertsEquals反过来equals方法为这些对象返回false。
这是堆栈跟踪,其中也包含对象的toString内容。
java.lang.AssertionError: expected: com.xyz.test.model.VerificationToken<VerificationToken(id=null, token=, user=User(userId=null, username=null, email=null, password=null, isActive=0, roles=null, imageLocation=null, enabled=false, isAccountNonLocked=true), expiryDate=Sat Dec 22 22:48:49 IST 2018)> but was: com.xyz.test.model.VerificationToken<VerificationToken(id=null, token=, user=User(userId=null, username=null, email=null, password=null, isActive=0, roles=null, imageLocation=null, enabled=false, isAccountNonLocked=true), expiryDate=Sat Dec 22 22:48:49 IST 2018)>
at org.junit.Assert.fail(Assert.java:88)
at org.junit.Assert.failNotEquals(Assert.java:834)
at org.junit.Assert.assertEquals(Assert.java:118)
at org.junit.Assert.assertEquals(Assert.java:144)
at ....
我对应的类如下。它具有lombak的@data批注,该批注生成了Equals方法。
@Entity
@Data
@AllArgsConstructor
@NoArgsConstructor
public class VerificationToken implements Serializable{
/**
*
*/
private static final long serialVersionUID = 8788934708305097680L;
private static final int EXPIRATION = 60 * 24;
@Id
private Long id;
private String token;
private User user;
private Date expiryDate;
public Date calculateExpiryDate(int expiryTimeInMinutes) {
Calendar cal = Calendar.getInstance();
cal.setTime(new Date());
cal.add(Calendar.MINUTE, expiryTimeInMinutes);
return new Date(cal.getTime().getTime());
}
}
我的用户类如下。这也有@Data Annotation,它会生成Equals方法
@Entity
@Table(name = "user")
@Data
public class User implements Serializable {
/**
*
*/
private static final long serialVersionUID = -6480423780555564185L;
@Id
private Integer userId;
private String username;
private String email;
@JsonIgnore
private String password;
private int isActive;
private Set<Role> roles;
private String imageLocation;
private boolean enabled;
private boolean isAccountNonLocked;
public User() {
super();
this.enabled=false;
this.isAccountNonLocked= true;
}
public User(User user) {
this.username = user.getUsername();
this.password = user.getPassword();
this.isActive = user.getIsActive();
this.roles = user.getRoles();
this.imageLocation=user.getImageLocation();
this.enabled= user.enabled;
this.isAccountNonLocked = user.isAccountNonLocked;
}
}
我的角色类如下:它也具有@Data Annotation,它会生成Equals方法
@Entity
@Table(name = "role")
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Role {
@Id
private Integer roleId;
private String role;
}
这是我的测试用例,其中包含对象实例化之一。
@Test
public void testCreateVerificationToken() {
User user = new User();
String token = new String();
VerificationToken myToken = new VerificationToken();
myToken.setToken(token);
myToken.setUser(user);
myToken.setExpiryDate(myToken.calculateExpiryDate(24*60));
ArgumentCaptor<VerificationToken> entityCaptor = ArgumentCaptor.forClass(VerificationToken.class);
when(mocktokenRepository.save(entityCaptor.capture())).thenReturn(myToken);
verificationTokenServiceImpl.createVerificationToken(user, token);
assertEquals(entityCaptor.getValue(),myToken);
}
其他对象实例化在此代码中发生。
@Override
public void createVerificationToken(User user, String token) {
VerificationToken myToken = new VerificationToken();
myToken.setToken(token);
myToken.setUser(user);
myToken.setExpiryDate(myToken.calculateExpiryDate(24*60));
tokenRepository.save(myToken);
}
我无法理解为什么两个对象被视为不相等。如果需要,我想提供更多详细信息。请帮忙。
谢谢。
答案 0 :(得分:5)
您要比较的两个对象在expiryDate
字段中具有不同的值。您正在使用new Date()
方法在代码中不同点的两个对象使用calculateExpiryDate
生成它们。他们将不平等。请注意,日期精度为毫秒。
我不建议您更改calculateExpiryDate
的实现以通过测试,而是可以将expiryDate
设置为硬编码的Date值。
答案 1 :(得分:3)
Date
具有毫秒精度,因此即使两个日期将相同的值打印到秒,equals
也可能返回false,因为它们相差毫秒。
您要使用的粒度大于毫秒。由于到期时间以分钟为单位,因此您可以将其用作最小时间单位,并将秒/毫秒设置为零。这是实现这一目标的快速方法。
public Date calculateExpiryDate(int expiryTimeInMinutes) {
Calendar cal = Calendar.getInstance();
cal.setTime(new Date());
cal.add(Calendar.MINUTE, expiryTimeInMinutes);
cal.set(Calendar.SECOND, 0);
cal.set(Calendar.MILLISECOND, 0);
return cal.getTime();
}
答案 2 :(得分:-2)
assertEquals
调用类equals
的{{1}}方法。您不会重写此方法,因此使用VerificationToken
的默认实现来检查实例是否相等(Object
)。如果a == b
不返回原始实例,但例如序列化或克隆的副本,该测试将失败。
如果您希望返回相同的实例,则应使用entityCaptor.getValue()
并修复导致返回不同实例的错误。如果您希望返回其他实例,则应首先执行assertSame
,在assertNotSame
中实现equals
和hashcode
,然后在{{1 }}。