JPA @Version字段不会递增

omjgkv6w  于 2022-11-14  发布在  其他
关注(0)|答案(6)|浏览(246)

我是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>

有什么不对吗?

7vhp5slm

7vhp5slm1#

这可能不完全是在主题上(因为我正在使用SpringData存储库),但当我在寻找我的问题的答案时,我最终来到了这里,所以这可能会帮助下一个人。
未返回以下和我的版本字段更新。

savedJpa = repository.save(updateJpa);

最后我的解决方案是

savedJpa = repository.saveAndFlush(updateJpa);
vfh0ocws

vfh0ocws2#

entityManager.persist()方法仅用于以前从未持久化的新实体。
因为你正在获取一个实体,所以你不需要调用persist或merge。脏检查将代表你进行更新。
无论如何,提交都会触发刷新,因此您应该看到更新。
确保使用javax.persistence.Version注解,而不是来自其他包(spring数据或类似数据)。

vu8f3i0k

vu8f3i0k3#

JPA仅在确实存在一些脏数据时才更新版本我假设学生195948的学期值已经为“1”。虽然您正在设置的值等于之前的值,理想情况下,实体状态没有变化,因此JPA不会更新版本值。(不是较早的版本),则实体的状态会发生变更,并相应地更新版本数据行。
谢了贾根

kb5ga3dv

kb5ga3dv4#

我可能会发现你的问题的一个重要点。尝试更换

private int version

private Long version

因为@Version标记可能无法与基本类型一起使用。

9lowa7mx

9lowa7mx5#

我遇到了一个不同的问题。我用JAX-RS开发REST API,并试图改变一个现有的资源,但是无论我点击PUT HTTP端点(更新数据的端点)多少次,@Version字段总是0。
调用更新HTTP终结点后,它将调用以下代码:
服务器代码:

public Todo save(Todo todo) {
    Todo createdTodo = this.entityManager.merge(todo);
    return createdTodo;
}

注意:如果实体尚不存在(如果实体的@Id不存在或ID为0),则Merge方法将创建新实体。如果实体的@Id存在,则它只使用新数据更新现有实体。
客户机正在发送以下json

{
  "todoTitle" : "testTitle",
  "priority" : 100 
}

我的错误:我忘了在JSON中发送ID,当我的Map将JSON数据放入Java对象时,它将“id”值设置为默认值0(因为primatives不能为空)。因此,每次我发送PUT HTPP请求(试图更新资源)时,我实际上是在创建一个新实体,而这个新实体的版本总是0,因为它是刚刚创建的。

u4vypkhs

u4vypkhs6#

当我们使用save()方法时,与保存操作相关的数据不会被刷新到DB中,除非显式调用flush()或commit()方法。
如果我们使用JPA实现(如Hibernate),则该特定实现将管理刷新和提交操作。
这里我们必须记住的一点是,如果我们决定自己刷新数据而不提交它,则更改对外部事务是不可见的,除非在此事务中进行了提交调用,或者外部事务的隔离级别为READ_UNCOMMITTED。

相关问题