23

假设我有这个类:

@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 方法做到这一点:

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

所以假设我有一个MyClass带有 name 和 surname 的对象实例:

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

听众没有听到这没关系,因为我只听更新。现在,如果我要像这样更新这个实例:

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

所以这会触发preUpdatemylistener 中的方法,当我尝试:

MyClass.findById(someLongId);

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

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

4

1 回答 1

30

我认为一种简单的方法是将先前的值保存在 JPA 不会持久的瞬态变量中。所以只需引入一个变量 previousSurname 并保存实际值,然后再在 setter 中覆盖它。

如果你想保存多个属性,如果你的类MyClassSerializable.

如果是这样,请添加一个后加载侦听器

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字段。我建议将其设置为包范围或使用包范围访问器并将 and 放在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;
     }

}
于 2013-10-11T08:46:49.907 回答