理解eneityManager的这三个方法的作用和区别,首先需要分清楚PersistenceContext 和 EntityManager.
PersistenceContext:是entity的一个实例。
EntityManager:是和PersistenceContext联系在一起的,被用来创建、删除或者查找一个持久化Entity实例。
换句话来说PersistenceContext可以说成是数据库的缓存。entitymanager的操作针对的都是整个实体,所以才叫实体管理器。
针对单个字段的操作使用createquery,createnativequery
hql,jpql,sql的使用不存在持久化上下文的缓存问题,即不适用session一级缓存???如果查到的结果封装到一个对象里则还是会在一级缓存中
1.merge
通过entityManager将一个存在的实体“同步到”persistenceContext中。
实体的状态将从其单独的状态转换为受persistenceContext管理的状态。
如果Entity是新创建的,则这个方法类似于persist()这个方法。
如果Entity已经存在的,则只作为更新操作。
E.merge()当实体对象O1为临时对象,会创建一个新对象O2,执行insert操作,并返回这个新对象,相当于S.saveOrUpdate()。此时O2为持久化对象,而O1仍然是游离对象。
E.merge()当实体对象O1位游离对象,即主键不为空:
首先查询缓存中是否有该主键对应的持久化对象,如果有,将缓存中的对象提取为O2,然后根据O1的值修改O2,并对O2执行update,返回O2
Hibernate不允许缓存中存在两个持久化对象对应同一个主键。
而JPA中不抛异常:
@Test
@Transactional
public void testJpa(){
User u1=entityManager.find(User.class, 1);
User u2=new User();
u2.setId(1);
u2.setUserName("Jack");
User u3=entityManager.merge(u2);
System.out.println(Arrays.asList(u1));
System.out.println(Arrays.asList(u2));
System.out.println(u1==u2);
System.out.println(u1==u3);
}
这是由于JPA不是在缓存中加载了第二个同一主键的实体对象,而是进行了实体对象的拷贝
merge 方法的主要作用是将用户对一个 detached 状态实体的修改进行归档,归档后将产生一个新的 managed 状态对象。
对不同状态下的实例 A , merge 会产生以下操作 :
1)如果 A 是一个 detached 状态的实体,该方法会将 A 的修改提交到数据库,并返回一个新的 managed 状态的实例 A2 ;
2)如果 A 是一个 new 状态的实体,该方法会产生一个根据 A 产生的 managed 状态实体 A2 ;
3)如果 A 是一个 managed 状态的实体,它的状态不会发生任何改变。但是系统仍会在数据库执行 UPDATE 操作;
4)如果 A 是一个 removed 状态的实体,该方法会抛出 IllegalArgumentException 异常
2.Flush
将PersistenceContext的信息同步到数据库中。
当触发Flush这个动作的时候,所有的实体都将会被insert/update/remove到数据库中。
数据库不会触发Commit的操作。
3.Refresh
Refresh的作用是从数据库中将Entity的状态进行更新操作。如果Entity和数据库中的数据不一致,将更新数据库中的数据到Entity中。即refresh操作只针对受托管状态(在persistencecontext中的对象)有效。
这种情况一般发 生在你获取了实体之后,有人更新了数据库中的记录,这时你需要得到最新的数据
1)如果 A 是一个 new 状态的实例,不会发生任何操作,但有可能会抛出异常,具体情况根据不同 JPA实现有关;
2)如果 A 是一个 managed 状态的实例,它的属性将会和数据库中的数据同步;
3)如果 A 是一个 removed 状态的实例,不会发生任何操作 ;
4)如果 A 是一个 detached 状态的实体,该方法将会抛出异常
????? 4. remove(Object entity)
remove 方法可以将实体转换为 removed 状态,并且在调用 flush() 方法或提交事物后删除数据库中的数据。
对不同状态下的实例 A , remove 会产生以下操作 :
1)如果 A 是一个 new 状态的实例, A 的状态不会发生任何改变,但系统仍会在数据库中执行 DELETE语句;
2)如果 A 是一个 managed 状态的实例,它的状态会转换为 removed ;
3)如果 A 是一个 removed 状态的实例,不会发生任何操作 ;
4)如果 A 是一个 detached 状态的实体,该方法将会抛出异常。
5.persist()
EntityManager没有save方法。
调用前的实体对象,如果主键使用setter设置了值,E.persist()会抛异常。而Session.save()不会抛异常,只是会忽略。
总结:
merger 将游离态修改并使之成为托管态,主要用于update操作,详见jpa update说明
flush 将托管态对象更新到数据库,但是没有执行commit,即没有提交事物
refresh 将托管态对象更新为db中的最新状态
persist 将临时态(即新new的对象,主键id为null)转化为托管态
remove 将托管态变为删除状态,即用该方法执行删除操作要先查库,再删除,如下
User user=em.find(User.class,"uuid");
em.remove(user);