Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行非常轻量级的封装,使得JAVA程序员可以随心所欲地使用对象编程思维来操作数据库。
Hibernate性能优化
第一,尽量降低访问数据库的次数
Cache:hibernate中有两次缓存,一级位于Session部分是必需的,第二级位于SessionFactory的外置缓存(SessionFactory缓存分两类,内置缓存和外置缓存,其中内置缓存存放映射元数据和预定义SQL语句,映射元数据是映射文件中数据的拷贝,而预定义SQL语句是在hibernate初始化阶段根据映射元数据推导出来的),二级缓存不是必需的,由开发人员自行指定,二级缓存能明显提高系统性能,但同时也会消耗内存,可通过配置文件来指定内存中能够加载的最多元素,有利于避免消耗过多内存。
在批量更新和批量删除时,通过绕过hibernate API,直接通过JDBC API来执行SQL语句,可以提高效率。
比如更新一万条记录
- tx = session.beginTransaction();
- Connection con = session.connection();
- PreparedStatement stmt = con.prepareStatement("update Customer set age=age+1 where age>0");
- tx.commit();
当然,使用存储过程效果更佳
- create or replace procedure batchUpdateCustomer(p_age in number) as
- begin
- update customer set age=age+1 where age>p_age;
- end;
- tx = session.beginTransaction();
- Connection con = session.connection();
- String procedure = "{call batchUpdateCustomer(?)}";
- CallableStatement cstmt = con.prepareCall(procedure);
- cstmt.setInt(1,0);
- cstmt.executeUpdate();
- tx.commit();
二级缓存的设置,第一种直接在hbm.xml中增加Cache元素,指明使用策略,如read-only,read-write等。
- <class name="" table="">
- <cache usage="read-write"/>
- ...........
- 配置echcache.xml
- <ehcache>
- <diskStore path="c://temp"/>
- <defaultCache
- maxElementsInMemory="10000"
- eternal="false"
- timeToIdleSeconds="120"
- timeToLiveSeconds="120"
- overflowToDisk="true"
- />
- <cache name="mypack.class"
- maxElementsInMemory="10000"
- eternal="false"
- timeToIdleSeconds="120"
- timeToLiveSeconds="120"
- overflowToDisk="true"
- />
- </ehcache>
第二适当的检索方式
延迟检索:对一些基本属性字段将其lazy设置为false,而对于一些可能需要消耗内存的字段,如clob,将其lazy设置为true,对于集合全部设为true。(class和set中有些属性)
不管是延迟检索还是立即检索都可以通过batch-size设置批量检索的数量。合理取值在3-10间(class和set元素中有些属性)。
第三避免无用的查询
使用投影查询,只查出实例的一部分字段
使用iterate()方法,在有些情况下轻微提高查询性能。因为iterate()首先检索ID字段,然后根据ID字段到hibernate的第一级缓存及二级缓存中查找匹配的对象,若存在,直接加入到检索结果集中,否则执行额外的select语句,根据ID字段到数据库中检索对象。
使用查询缓存,这样第一次执行查询语句时,会把查询结果放在第二级缓存中,注意二级缓存中存放的只是实体的OID,对投影查询,会存放所有的数据值。
启用查询缓存步骤:
1.配置二级缓存
2.在hibernate的hibernate.properties配置文件中设置查询缓存属性;
hibernate.cache.use_query_cache=true
3.Query query = session.createQuery("from customer c where c.age>:age");
query.setInteger("age",age);
query.setCacheable(true);
第三配置
使用连接池