我是JPA和Hibernate的新手,我在乐观锁定方面遇到了一个问题。我有一个类,它有一个@Version注解字段。当我更新这个类表示的实体时,版本计数器没有增加。下面是我的代码:类:
@Entity
@Table (name = "Studenten")
public class Student implements Serializable{
private static final long serialVersionUID = 705252921575133272L;
@Version
private int version;
private int matrnr;
private String name;
private int semester;
public Student (){
}
public Student (int matrnr, String name){
this.matrnr = matrnr;
this.name = name;
}
public Student (int matrnr, String name, int semester){
this(matrnr, name);
this.semester = semester;
}
下面是主要的方法:
public static void main(String[] args) {
EntityManagerFactory emf = Persistence.createEntityManagerFactory("VEDA_Vortrag");
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
try{
tx.begin();
Student s = em.find(Student.class, 195948);
s.setSemester(1);
em.persist(s);
tx.commit();
}catch(Exception e){
if(tx != null && tx.isActive()){
tx.rollback();
System.out.println("Error in Transaction. Rollback!");
}
}
finally{
em.close();
emf.close();
}
}
下面是控制台显示的内容:
Hibernate:
select
student0_.matrnr as matrnr0_0_,
student0_.name as name0_0_,
student0_.semester as semester0_0_,
student0_.version as version0_0_
from
Studenten student0_
where
student0_.matrnr=?
Hibernate:
update
Studenten
set
name=?,
semester=?,
version=?
where
matrnr=?
谁能告诉我怎么了?
**编辑:**好的,我已经尝试了一些方法。我已经将锁定设置为LockModeType.OPTIMISTIC_FORCE_INCREMENT,并收到以下错误消息:
ERROR: HHH000099: an assertion failure occurred (this may indicate a bug in Hibernate, but is more likely due to unsafe use of the session): org.hibernate.AssertionFailure: cannot force version increment on non-versioned entity
Jun 20, 2014 9:00:52 AM org.hibernate.AssertionFailure <init>
所以很明显,我有一个非版本化的实体。但是为什么呢?我在我的类中有@Version注解。
**编辑:**我开始认为问题出在persistence.xml中。
<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
version="2.0">
<persistence-unit name="VEDA_Vortrag">
<provider>org.hibernate.ejb.HibernatePersistence</provider>
<properties>
<property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost:3306/uni"/>
<property name="javax.persistence.jdbc.user" value="*******"/>
<property name="javax.persistence.jdbc.password" value="******"/>
<property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
<property name="hibernate.dialect" value="org.hibernate.dialect.MySQLDialect"/>
<property name="hibernate.show_sql" value="true"/>
<property name="hibernate.format_sql" value="true"/>
</properties>
</persistence-unit>
有什么不对吗?
6条答案
按热度按时间7vhp5slm1#
这可能不完全是在主题上(因为我正在使用SpringData存储库),但当我在寻找我的问题的答案时,我最终来到了这里,所以这可能会帮助下一个人。
未返回以下和我的版本字段更新。
最后我的解决方案是
vfh0ocws2#
entityManager.persist()
方法仅用于以前从未持久化的新实体。因为你正在获取一个实体,所以你不需要调用persist或merge。脏检查将代表你进行更新。
无论如何,提交都会触发刷新,因此您应该看到更新。
确保使用
javax.persistence.Version
注解,而不是来自其他包(spring数据或类似数据)。vu8f3i0k3#
JPA仅在确实存在一些脏数据时才更新版本我假设学生195948的学期值已经为“1”。虽然您正在设置的值等于之前的值,理想情况下,实体状态没有变化,因此JPA不会更新版本值。(不是较早的版本),则实体的状态会发生变更,并相应地更新版本数据行。
谢了贾根
kb5ga3dv4#
我可能会发现你的问题的一个重要点。尝试更换
至
因为@Version标记可能无法与基本类型一起使用。
9lowa7mx5#
我遇到了一个不同的问题。我用JAX-RS开发REST API,并试图改变一个现有的资源,但是无论我点击PUT HTTP端点(更新数据的端点)多少次,@Version字段总是0。
调用更新HTTP终结点后,它将调用以下代码:
服务器代码:
注意:如果实体尚不存在(如果实体的@Id不存在或ID为0),则Merge方法将创建新实体。如果实体的@Id存在,则它只使用新数据更新现有实体。
客户机正在发送以下json
我的错误:我忘了在JSON中发送ID,当我的Map将JSON数据放入Java对象时,它将“id”值设置为默认值0(因为primatives不能为空)。因此,每次我发送PUT HTPP请求(试图更新资源)时,我实际上是在创建一个新实体,而这个新实体的版本总是0,因为它是刚刚创建的。
u4vypkhs6#
当我们使用save()方法时,与保存操作相关的数据不会被刷新到DB中,除非显式调用flush()或commit()方法。
如果我们使用JPA实现(如Hibernate),则该特定实现将管理刷新和提交操作。
这里我们必须记住的一点是,如果我们决定自己刷新数据而不提交它,则更改对外部事务是不可见的,除非在此事务中进行了提交调用,或者外部事务的隔离级别为READ_UNCOMMITTED。