我有在@Entity
中用Java
注释的数据库类。
我想比较要按字段比较字段的2个类实例,但从Hamcrest
收到对象不相等的错误。除非我在equals
中自动生成hashCode
和Eclipse
,否则它将持续存在。
实体的结构如下:
@Entity
@Table(name = "book")
public class Book extends BaseEntity {
@NotNull
@Column
private String title;
}
和
@MappedSuperClass
class BaseEntity {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
}
为了简单起见,我省略了所有的getter和setter方法。
我上面如何使用实体的示例代码:
public void updateBookTitles(bookIds) {
List<Book> books = bookRepository.findByBookIds(bookIds);
books.forEach((book) -> {
book.setTitle("Example: " + book.getTitle());
});
bookRepository.saveAll(books);
}
测试时:
@Test
public void updateSingleBookTitle() {
Book book = new Book();
book.setId(1L);
book.setTitle("Stack Overflow manual");
when(bookRepository.findByBookIds(Arrays.asList(1L))).thenReturn(Arrays.asList(book));
updateBookTitles(Arrays.asList(1L));
Book expectedBook = new Book();
expectedBook.setId(1L);
expectedBook.setTitle("Stack Overflow manual");
verify(bookRepository, times(1)).saveAll(Arrays.asList(expectedBook))
}
因此,我有问题。除了自动生成
equals
和hashCode
以外,还有什么方法可以使对象在测试过程中相等,并且在存储到数据库时不会降低性能。据我所知,目前对象是由id
字段而不是所有字段检查的。 最佳答案
您可以使用samePropertyValuesAs
方法:
Book book1 = new Book();
Book book2 = new Book();
assertThat(book1, samePropertyValuesAs(book2));
这将逐个字段地比较您的对象,并在它们彼此相等时执行断言。如果您需要忽略某些字段,请查看Shazamcrest - Reusable Hamcrest matchers suitable for automated testing。
更新1:
尝试使用下一个代码:
verify(bookRepository, times(1)).saveAll(argThat(new ArgumentMatcher<List<Book>>() {
@Override
public boolean matches(Object argument) {
List<Book> bookToCheck = (List<Book>) argument;
assertThat(bookToCheck.get(0), sameAsBean(expectedBook));
return true;
}
}));
更多信息:ArgumentMatcher