java – 乐观锁定和org.hibernate.StaleObjectStateException:

我只是在尝试乐观锁定.

我有以下课程:

@Entity
public class Student {

    private Integer id;
    private String firstName;
    private String lastName;
    private Integer version; 
@Version
    public Integer getVersion() {
        return version;
    }

//all other getters ommited.
}

现在我正在抓住其中一个学生并尝试同时更新其属性.

Thread t1 = new Thread(new MyRunnable(id));
    Thread t2 = new Thread(new MyRunnable(id));
    t1.start();
    t2.start();

和MyRunnable内部:

public class MyRunnable implements Runnable {
    private Integer id;
    @Override
    public void run() {
        Session session = HibernateUtil.getSessionFactory().openSession();       
        session.beginTransaction();
        Student student = (Student) session.load(Student.class,id);
        student.setFirstName("xxxx");
        session.save(student);
        session.getTransaction().commit();
        System.out.println("Done"); 
    }

    public MyRunnable(Integer id){
        this.id = id;
    }
}

一个事务成功更新对象和第二个事务抛出的情况:

org.hibernate.staleobjectstateexception: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect): [com.vanilla.entity.Student#1]

还行吧.

我的问题是:
1)如果我希望第二个事务什么也不做,并且不抛出任何异常,我该怎么办?

2)如果我希望第二个事务覆盖第一个事务更新的数据,我该怎么办?

谢谢.

解决方法

我试着回答你的问题:

>您使用乐观锁定.因此,您希望在版本冲突时抛出OptimisticLockException – 但您可以捕获它并且不执行任何操作.您不能将其关闭一秒(无论这意味着什么)事务,因为您不知道是否会发生版本冲突(这是乐观锁策略的本质:乐观假设是版本冲突不会经常发生)
>如果发生OptimisticLockException,您基本上有两个选项:

>放弃更改(也许刷新当前状态)
>刷新您的实体(实体)的版本并尝试再次提交

如果您有并发更新,问题是如何或谁将决定哪个状态是实际的“正确”(表示最新的).只要保证一致性,我就不在乎了.

相关文章

最近看了一下学习资料,感觉进制转换其实还是挺有意思的,尤...
/*HashSet 基本操作 * --set:元素是无序的,存入和取出顺序不...
/*list 基本操作 * * List a=new List(); * 增 * a.add(inde...
/* * 内部类 * */ 1 class OutClass{ 2 //定义外部类的成员变...
集合的操作Iterator、Collection、Set和HashSet关系Iterator...
接口中常量的修饰关键字:public,static,final(常量)函数...