获取对象字段前值hibernate JPA

时间:2013-10-11 08:28:53

标签: java hibernate caching jpa annotations

我们假设我有这门课程:

@EntityListeners({MyListener.class})
class MyClass {
  String name;
  String surname;

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name; 
  }

  public String getSurname() {
    return name;
  }

  public void setSurname(String name) {
    this.name = name; 
  }

  public void save() {
    JPA.em().persist(this);
    return this;
  }

  public void update() {
    JPA.em().merge(this);
  }

  public static MyClass findById(Long id) {
    return JPA.em().find(MyClass.class, id);
  }
}

现在在我的MyListener类中,我试图找出先前的值MyClass实例与即将保存(更新)到数据库的新值。我用preupdate metdhod执行此操作:

@PreUpdate
public void preUpdate(Object entity) {
...some logic here...unable to get the old object value in here           
}

假设我有一个名称和姓氏为MyClass的对象实例:

MyClass mycls = new MyClass();
mycls.setName("Bob");
mycls.setSurname("Dylan");
mycls.save();

听众没有接受这个,因为我只是在听更新。 现在,如果我像这样更新这个实例:

MyClass cls = MyClass.findById(someLongId)
cls.setSurname("Marley");
cls.update();

因此,这会触发mylistener中的preUpdate方法,当我尝试:

MyClass.findById(someLongId);

当我调试时,我已经获得了新更新的实例,但更新尚未发生,因为当我在列姓氏中检入数据库时​​,它仍然是Dylan

如何从我的preUpdate方法中获取数据库中的值,而不是我刚更新的那个?

1 个答案:

答案 0 :(得分:21)

我认为一种简单的方法是将以前的值保存在JPA不会持久的瞬态变量中。 因此,只需引入变量previousSurname并在设置器中覆盖它之前保存实际值。

如果您要保存多个属性,那么您的课程MyClassSerializable.

会很容易

如果是这样,添加一个post load listener

public class MyClass implements Serializable {

     @Transient
     private transient MyClass savedState;

     @PostLoad
     private void saveState(){
        this.savedState = SerializationUtils.clone(this); // from apache commons-lang
     }

}

但请注意,savedState是一个分离的实例。

然后,您可以访问EntityListener中的上一个状态。

您还可以将PostLoad侦听器移至EntityListener类。但是,您需要访问savedState字段。我建议使用包作用域或使用包范围访问器,并将MyClassMyListener放在同一个包中,

public class MyListener {

    @PostLoad
    private void saveState(MyClass myClass){
         myClass.saveState(SerializationUtils.clone(myClass)); // from apache commons-lang
    }

}

public class MyClass implements Serializable {

     @Transient
     private transient MyClass savedState;

     void saveState(MyClass savedState){
        this.savedState = savedState;
     }

}