10.7. 自动状态检测

优质
小牛编辑
148浏览
2023-12-01

Hibernate 的用户曾要求一个既可自动分配新持久化标识(identifier)保存瞬时(transient)对象,又可更新/重新关联脱管(detached)实例的通用方法。saveOrUpdate() 方法实现了这个功能。

// in the first session
Cat cat = (Cat) firstSession.load(Cat.class, catID);

// in a higher tier of the application
Cat mate = new Cat();
cat.setMate(mate);

// later, in a new session
secondSession.saveOrUpdate(cat);   // update existing state (cat has a non-null id)
secondSession.saveOrUpdate(mate);  // save the new instance (mate has a null id)

saveOrUpdate() 用途和语义可能会使新用户感到迷惑。首先,只要你没有尝试在某个 session 中使用来自另一 session 的实例,你就应该不需要使用 update()saveOrUpdate(),或 merge()。有些程序从来不用这些方法。

通常下面的场景会使用 update()saveOrUpdate()

  • 程序在第一个 session 中加载对象

  • 该对象被传递到表现层

  • 对象发生了一些改动

  • 该对象被返回到业务逻辑层

  • 程序调用第二个 session 的 update() 方法持久这些改动

saveOrUpdate() 做下面的事:

  • 如果对象已经在本 session 中持久化了,不做任何事

  • 如果另一个与本 session 关联的对象拥有相同的持久化标识(identifier),抛出一个异常

  • 如果对象没有持久化标识(identifier)属性,对其调用 save()

  • 如果对象的持久标识(identifier)表明其是一个新实例化的对象,对其调用 save()

  • 如果对象是附带版本信息的(通过 <version><timestamp>)并且版本属性的值表明其是一个新实例化的对象,save() 它。

  • 否则 update() 这个对象

merge() 可非常不同:

  • 如果 session 中存在相同持久化标识(identifier)的实例,用用户给出的对象的状态覆盖旧有的持久实例

  • 如果 session 没有相应的持久实例,则尝试从数据库中加载,或创建新的持久化实例

  • 最后返回该持久实例

  • 用户给出的这个对象没有被关联到 session 上,它依旧是脱管的