最近在编写代码的时候遇到一个小坑,最后查询了一下发现是自己没考虑到JPA的四种状态(哈哈哈,这里当时想给自己一个脑瓜子,的确有一点点短路了),所以这里总结一下我自己对Jpa四种状态的理解!
ok,废话不多说,上才艺;
首先JPA的四种状态
临时状态:刚创建出来,没有与entityManager发生关系,没有被持久化,不处于entityManager中的对象;
通俗举例子来说就是你NEW了一个对象,但是你没有对其就行操作,就比如:User user= New User();这就是一个临时状态;
持久状态:与entityManager发生关系,已经被持久化,您可以把持久化当作实实在在的数据库记录;
通俗一点就是new出来的对象对数据库进行了操作,就比如刚刚new的user对象,进行user.find()操作,从数据库中查询了数据;这就是一个持久化状态,当然user不一定要是find操作,也可以是其它的数据操作;
说到这持久化有个地方要知道一下,不然容易踩坑:
在实例处于持久状态时,如果你修改了数据,使用了set方法,然后你commit操作后是会同步数据的,也就是你set里的数据会同步到数据库里;
删除状态:执行remove方法,事物提交之前;
就是删除数据嘛!
对于这个删除也还有一个坑需要特别注意 (如图),这是进行了find(查询)、remove(删除)操作和persist(插入)操作,但实际上其实只进行了find操作,这是因为持久状态下的进行删除时如果再进行插入就会进行回归,就是不再执行删除,回到查询操作;
游离状态:游离状态就是提交到数据库后,事物commit后实体的状态,因为事物已经提交了,此时实体的属性任你如何改变,也不会同步到数据库,因为游离已经是没人管的孩子了,不在持久化上下文中;
通俗说就是你的对象被创建后没有与entityanage发生过关系(也就是没有过像find()的这种操作),这就是就是游离状态了;
下面这么一个图可能比较直观一点:
另外最后说明几点容易被坑的地方:
public void remove(Object entity)
remove力法可以将实体转换为removed状态,并且在调用flush()方法或提交事物后删除数据库中的数据,对不同状态下的实例A.remove会产生以下操作:
1 如果A局一个new状态的实例,A的状态不会发生任何改变,但系纷仍会在数拗南中执行DELETE语句; 2.如果A是一个managed(持久)状态的实例,它的状态会转换为removed;
3.知果A是一个removed(删除)状态的实例,不会发生任匀找作;
4.如果A是一个datached(游离)状态的实,该方法将会跑出异常;
public void refresh(Object entity)
refresh方法可以保证当前的实例与数据库中的实例的内容一致。对不同状态下的实例A. refresh会产生以下操作:
1.如果A是一个new状态的实例,不会发生任何操作,但有可能会抛出异常,具体根据不同的JPA实现有关; 2如果A是一个managed状态的实例,它的属性将会和效据库中的数据同步;
3.如果A是一个romoved状态的实例,该方法将会抛出异常: Entity not managed; 4.如果A是一个datached状态的实例,该对法将会抛出异常。
public void persist(Object entity)
persist方法可以将实例转换为managed(托管)状态。在调用flush()方法或提交事物后,实例将会被插入到数据库中.对不同状态下的实例A. persist会产生以下操作:
1.如果A是一个new状态的实体,它将会转为managed状态;
2.如果A是一个managed状态的实体。它的状态不会发生任何改变。但是系统仍会在数据据库执行INSERT操作; 3.出果A是一个removed(删除)状态的实体,它将会转换为受控状态;
4.如果A是一个datached(游离)状态的实体,该方法会抛出T11egalkngumontioxseprion异常,具体异常根据不同的JPA实现有关。