Session.find()方法

本文探讨了Hibernate中find与iterate方法的区别及应用场景。find方法通过单条SQL实现查询,无法利用缓存;iterate方法先查询符合条件记录的ID,再利用缓存减少数据库访问。文章还介绍了iterate结合evict方法的应用,以解决大数据量时的内存溢出问题。
查询性能往往是系统性能表现的一个重要方面,查询机制的优劣很大程度上决定了系统的整体性能。这个领域往往也存在最大的性能调整空间。

hibernate2中Session.find()对应于3中的session.createQuery().list();
hibernate2中Session.iterate()对应于3中的session.createQuery().iterate();
find和iterate区别:
find方法通过一条Select SQL实现了查询操作,而iterate方法要执行多条Select SQL.
iterate第一次查询获取所有符合条件的记录的id,然后再根据各个id从库表中读取对应的记录,这是一个典型的N+1次的查询问题,如果符合条件记录有10000条,就需要执行10001条Select SQL,可想性能会如何的差。

那为什么要提供iterator方法,而不只是提供高效率的find方法?

原因1.与hibernate缓存机制密切相关
find方法实际上是无法利用缓存的,它对缓存只写不读。
find方法只执行一次SQL查询,它无法判断缓存中什么样的数据是符合条件的,也无法保证查询结果的完整性。而iterate方法,会首先查询所有符合条件记录的id,然后根据id去缓存中找,如果缓存中有该id,就返回,没有可以根据id再去数据库查询。
String hql = "from TUser where age > ?";
List userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
Iterator it = session.iterate(hql, new Integer(18), Hibernate.INTEGER);
顺序执行,iterate方法只会执行一次SQL查询,就是查找id,然后根据id就可以从缓存中获得数据。

String hql = "from TUser where age > ?";
List userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
userList = session.find(hql, new Integer(18), Hibernate.INTEGER);
缓存是不起作用的。
如果目标数据读取相对较为频繁,通过iterate这种机制,会减少性能损耗。

原因2.内存使用上的考虑
find方法将一次获得的所有记录并将其读入内存。如果数据量太大,可能会触发OutOfMemoryError,从而导致系统异常。解决方案之一就是结合iterate方法和evict方法逐条对记录进行处理,将内存消化保持在一个可以接受的范围之内。如:
String hql = "from TUser where age > ?";
Iterator it = session.iterate(hql, new Integer(18), Hibernate.INTEGER);
while(it.hasNext()) {
TUser user = (TUser)it.next();

//将对象从一级缓存中删除
session.evict(user);

//二级缓存可以设定最大缓存量,达到后自动对较老数据进行废除,但也可以通过编
//码移除,这样有助于保持数据有效性。
sessionFactory.evict(TUser.class, user.getID());
}
这段代码是使用 MyBatis 进行数据库操作的基本代码,主要步骤包括创建 `SqlSessionFactory`、打开 `SqlSession`、执行查询操作、关闭 `SqlSession` 并返回查询结果。不过,代码存在一些可以优化的地方,以下是优化建议和优化后的代码: ### 优化点 1. **资源管理**:使用 `try-with-resources` 语句来确保 `SqlSession` 资源在使用完毕后能自动关闭,避免资源泄漏。 2. **异常处理**:添加异常处理机制,捕获并处理可能出现的异常,增强代码的健壮性。 3. **日志记录**:在出现异常时记录日志,方便后续排查问题。 ### 优化后的代码 ```java import org.apache.ibatis.session.SqlSession; import org.apache.ibatis.session.SqlSessionFactory; import org.apache.ibatis.session.SqlSessionFactoryBuilder; import java.io.InputStream; import java.util.List; // 假设 User 类是用户实体类 class User { // 类的属性和方法 } public class MyBatisExample { public List<User> findAllUsers(InputStream resourceAsStream) { List<User> list = null; try (SqlSession session = new SqlSessionFactoryBuilder().build(resourceAsStream).openSession()) { list = session.selectList("usermapper.findAll"); } catch (Exception e) { // 这里可以使用日志框架记录异常信息,例如使用 SLF4J e.printStackTrace(); } return list; } } ``` ### 解释 - `try-with-resources` 语句会在代码块执行完毕后自动调用 `SqlSession` 的 `close` 方法,确保资源被正确关闭。 - `catch` 块捕获可能出现的异常,并打印异常堆栈信息,方便调试。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值