Hibernate constraint ConstraintViolationException. Is there an easy way to ignore duplicate entries?
My question is whether there is an easy way to make Hibernate to just ignore inserts which would in effect insert duplicates?
How could Hibernate possibly know that a record has a non unique value without actually inserting the record?
If you are doing batch inserts and don't want to rollback the whole transaction and discard the session in case of a ConstraintViolationException
(that's what you should do in theory after an exception, see this thread and this previous answer), my suggestion would be to use the StatelessSession
API and to catch the ConstraintViolationException
.
Hibernate can't do this. You can however do it yourself. Essentially there are two options:
- Check for duplicates before insert; or
- Catch the ConstraintViolationException and then make sure that the violation is due to a duplicate (since there can be other reasons, like null fields).
The former has the disadvantage that you are checking for duplicates on every insert when the likelihood of there actually being a duplicate may be low. The latter will be faster for usual case where no duplicates occur, but has the disadvantage that you need to check for duplicates after the face. When a ConstraintViolationException is thrown it invalidates the current session; this means you will need to flush before doing a search of a duplicate.
Checking for duplicates before insert is probably the cleanest approach unless there is a major performance problem that you need to worry about. Make sure you do the lookup and insert in a transaction to ensure that someone doesn't add a duplicate between the lookup and insert, otherwise you will get a ConstraintViolationException.
You can use @SqlInsert annotation on your entity, to change the sql hibernate uses to make insertion in Database, hibernate currently must be using this statement
insert into method(name, id) values (?,?)
If you are using mysql, you can use the @SqlInsert statement to change the insert sql by using insert ignore mysql statement
@SQLInsert(sql = "insert ignore into method(name, id) values (?,?)")
However you should get the insert sqls by enabling show sql mode in hibernate, since the order of the columns is generated via some logic, and their docs suggest this way.