我有在@Entity中用Java注释的数据库类。

我想比较要按字段比较字段的2个类实例,但从Hamcrest收到对象不相等的错误。除非我在equals中自动生成hashCodeEclipse,否则它将持续存在。

实体的结构如下:

@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))
}


因此,我有问题。除了自动生成equalshashCode以外,还有什么方法可以使对象在测试过程中相等,并且在存储到数据库时不会降低性能。据我所知,目前对象是由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

07-28 01:05
查看更多