What is ‘Dirty Check’? 🤔
Hibernate checks the entity objects that are in the ‘managed’ status, that is, they have been added to the persistence context. When an entity object is loaded, a copy of all properties of that entity object is created. At the time of synchronization, which we call flush time, the properties of the entity object are matched with the properties of the loaded object and the difference is checked. This process is called “Hibernate Dirty Check”.
Let’s start with a basic entity.
@Entity
@Table(name="comment")
@DynamicUpdate
public class CommentEntity{
@Id
@GeneratedValue(strategy=GenerationType.SEQUENCE)
private Long id;
private String text;
private String owner;
}
Enter fullscreen mode Exit fullscreen mode
and persist the created data to database via EntityManager.
@Repository
@Transactional
public class CommentRepository {
@Autowired
EntityManager entityManager;
public void save() {
CommentEntity commentEntity= new CommentEntity();
commentEntity.setText("my comment");
commentEntity.setOwner("my owner");
entityManager.persist(commentEntity);
}
}
Enter fullscreen mode Exit fullscreen mode
When we call the save method of the bean named CommentRepository, it will insert into the database. The SQL output is as follows.
Hibernate: call next value for hibernate_sequence
Hibernate: insert into comment (id, text, owner) values (?, ?, ?)
Enter fullscreen mode Exit fullscreen mode
What if we try to persist the same data to our database after we get it?
public void update() {
CommentEntity commentEntity =
entityManager.find(CommentEntity.class, 1l);
entityManager.persist(commentEntity);
}
Enter fullscreen mode Exit fullscreen mode
Here is the output that only makes a select query by hibernate
Hibernate: select commententity0_.id as id1_1_0_, commententity0_.text
as text2_1_0_, commententity0_.owner
as owner3_1_0_ from comment commententity0_
where commententity0_.id=?
Enter fullscreen mode Exit fullscreen mode
Well… What if we change the property of the object and persist again?
public void update() {
CommentEntity commentEntity = entityManager
.find(CommentEntity.class, 1l);
commentEntity.setText("my other text");
entityManager.persist(commentEntity);
}
Enter fullscreen mode Exit fullscreen mode
Now, Hibernate will apply update action after select
Hibernate: select commententity0_.id as id1_1_0_,
commententity0_.text as text2_1_0_, commententity0_.owner
as owner3_1_0_ from comment commententity0_ where commententity0_.id=?
Hibernate: update comment set text=? where id=?
Enter fullscreen mode Exit fullscreen mode
Each time, Hibernate checks the properties of the entity object with the last loaded snapshot. If there is a change, it performs the update.
That’s why this situation is called “Hibernate Dirty Check”. In fact, it causes a performance loss.
What is the solution? 🧠
When an entity is loaded, the ‘Hibernate dirty checking mechanism’ compares the snapshot of the current entity with the loaded entity, but we can also turn off this comparison when we are not going to update it.
@Transactional(readOnly = true)
Enter fullscreen mode Exit fullscreen mode
If we mark the method we are processing as readOnly = true, there will be no ‘Hibernate dirty check’ operation, as there will be no update operation. This gives us performance.
暂无评论内容