使用jpa的时候set实体类属性自动持久化的解决方案
使用jpa的时候set实体类属性自动持久化
实例代码
Set<User> users = new HashSet<User>(); User user = null; for(int i = 0; i < 10; i++){ user = new User(); user.setUserName("wy" + i); users.add(user); } Company company = userDao.getCompany(); company.setUsers(users);
当使用了实体类set属性的时候,但是我们并没有持久化,却自动保存到数据库了。
原因
Hibernate分为三种基本的状态:游离态、自由态(临时状态)、持久态。
持久化状态:与session关联并且和在数据库有数据,已经持久化了并且在数据库的缓存当中了。
我这里的这个对象应该是持久化状态的对象然后我直接构造了一个user对象的set集合,同时对这个对象进行set操作,那么缓存Session中的数据发生改变,那么接着数据库也会跟着进行相应的改变。所以就执行了update的更新操作。
解决办法:
1.如果这个对象(例子中的company)本身不需要用的话,可以直接new一个Company的对象出来然后再setUsers这个时候因为不是Session中的数据,那么不会因为对象的属性发生改变而同步到数据库中去。
2. 如果这个对象(例子中的company)要用的到,那么,在set之前可以先将其转为游离态,这样的话会用到session的几个方法:close、clear、evict。
close方法:关闭session这样这个对象肯定是游离态了,因为session已经关闭了,但是往往我们实际的开发过程中,session在后面是要用的到的,所以这个方法可行,但是不一定用得上,分清具体的情况。
clear方法:将session中的所有的对象全部清除出缓存,这个方式有点劳师动众,不过session清除了全部的对象之后自然就会变为游离态了,这样做不是很好吧我感觉。
evict方法:将某一个对象清除出缓存session,这个方法是很好的实现方式,推荐使用。调用的时候是这样的,session.evict(Object obj);这样就可以了。
使用注解获取session方法
@PersistenceContext private EntityManager entityManager;
然后在方式中我们使用如下方式获取session
HibernateEntityManager hEntityManager = (HibernateEntityManager)entityManager; Session session = hEntityManager.getSession();
然后使用获取到的session根据上面说的操作对象既可
jpa实体类一对多set与list使用
当从一的一端取出其所对应的多的一端时,如果用的是set那么取出多的一端的值时顺序是无序的,如果用的是list那么取出多的一端的值时顺序是有序的(其实就是list与set的特性罢了,然鹅。。。。。。)
问题:因为set查询出的数据是无序的,如果当一的一端对应5条数据,而分页的单页只显示3条数据,当到下一页时,其中的两条数据可能与与上一页的3条数据出现随机相同的情况,因此会导致数据显示的缺失和无序。
注:不过jpa设计的时候用的是set不是没有道理的,主要是利用set的数据不可重复性,用于避免数据的重复,比如新增数据的时候避免数据的重复插入。
完美的解决办法:
在实体类set属性上加上@orderBy("id asc")属性进行排序,然后在取值的时候使用LinkedHashSet(不可重复且有序sss)接住即可
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
相关文章
详解SpringBoot中@SessionAttributes的使用
这篇文章主要通过示例为大家详细介绍了SpringBoot中@SessionAttributes的使用,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下2022-07-07解决@RequestBody使用不能class类型匹配的问题
这篇文章主要介绍了解决@RequestBody使用不能class类型匹配的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2021-07-07
最新评论