您的位置:首页 > 其它

EJB的persist 和 merge,refresh

2015-09-22 01:34 162 查看
http://home.51.com/kufei520/diary/item/10042994.html

4. EntityManager接口

find(Class<T> entityClass, Object primaryKey):<T> T //根据主键查找,没有符合条件记录,返回null ,相当于Hibernate:get();

getReference(Class<T> entityClass,Object primaryKey):<T>T //根据主键查找,没有符合条件记录,抛出异常EntityNotFoundException 相当于 Hibernate:load();

persist(Object entity):void //将实体变成受管态,并持久化它,将在数据库中增加一条记录 相当于Hibernate:save();

merge(T entity):T //合并一个实体到持久化上下文中,将更新数据库记录,并返回持久化对象 相当于 Hibernate:merge();

remove(Object entity):void //删除一个实体,将删除数据库记录

clear():void //清除持久化上下文,将导致托管态对象变成游离态对象

contains(Object entity):boolean //检查一个实体对象是否包在持久化上下文中

close(); //关闭持久化上下文

flush(); //对象同步到数据库,有两种同步方式

FlushModeType.AUTO //事务提交和执行查询时同步

FlushModeType.COMMIT; //只有事务提交时同步

setFlushMode(FlushModeType flushMode) //设置同步方式

refresh(); //数据库同步到对象 (数据库 -->对象)

getTransaction().begin(); //开启事务

getTransaction().commit(); //提交事务,事务结束

5. 持久化上下文和实体的生命周期

新建 -> 托管(受管) -> 脱管(游离) -> 删除

1) 新建new:new Instance persist(); 新建一个实体对象,还未持久化,没有持久化标志

2) 托管managed:数据库里和持久化上下文中都有 实体对象

3) 游离detached:持久化上下文中没有,但是又具有持久化标志(oid) 可调用merge(),不能调用persist();

4) 删除removed:hibernate没有(数据库中和持久化上下文中都有,但是通过remove()标志实体是被删除的)

总结:

1) 不调用persist, 而是调用merge持久保存entity对象

2) 在事务当中(ejb默认一个方法是一个事务)或者持久化上下文中,先查询entity, 再进行操作

3) 调用find方法, 不要调用getReference()

merge()也有persist()的作用!

persist会把传进去的实体放到持久化上下文中,此时如果持久化上下文中有了这个实体,就会抛出javax.persistence.EntityExistsException,没有的话事务提交的时候把那个对象加进数据库中,如果数据库中已经存在了那个对象(那一行),就会抛出com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException;

而merge会在持久化上下文中生成传进去的实体的受管版本,如果已经有了受管版本,那也不会抛出异常,然后把那个受管的实体返回出来,事务提交的时候如果数据库中不存在那个对象(那一行),就把把那个受管的加进去,存在的话就替换掉原来的数据。merge是如果持久化上下文中有了受管版本,那就更新,没有就复制一份,返回受管的。

再次总结persist(①,②-③,④-⑤):

(这里说的抛出的异常都是指对象(或者数据库中的行)重复的异常)

① 如果persist的是一个受管实体(即已经在上下文中),就不会抛出异常。
②如果persist的是一个游离实体(即上下文中没有它),而上下文中又没有它的受管版本,数据库中也没有,也不会抛出异常,而会把这个实体写进数据库中。
③如果persist的是一个游离实体(即上下文中没有),而上下文中又没有它的受管版本,数据库却有这个实体,那么EntityManager在persist它的时候不会抛出异常,但是事务提交的时候就会抛出异常:
Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException: Duplicate entry '7' for key 1;
④如果persist的是一个游离实体(即上下文中没有),而上下文中却有它的受管版本,数据库中又没有这个实体,那么还是不会抛出异常,而是把它的受管版本加进去(不是那个游离的,是那个受管的!)(即,这种情况persist和没persist是一样的!)。
⑤如果persist的是一个游离实体(即上下文中没有),而上下文中却有它的受管版本,数据库中也有了这个实体,那么EntityManager在persist它的时候就会抛出异常:javax.persistence.EntityExistsException

而merge就不会抛出什么对象重复的异常的了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: