缓存
缓存的作用主要用来提高性能,可以简单的理解为一个Map;使用缓存涉及到三个操作:把数据放入缓存、从缓存中取出来、删除缓存中的无效数据;
Hibernate把事务交给了第三方的缓存框架来处理
一级缓存:(session级别)
一级缓存在hibernate中是不要配置的,一级缓存是放在session中的,save,update,saveOrUpdate load ,get ,list ,iterate,lock这些方法都会将对象放在一级缓存中,一级缓存不能控制缓存的数量,所以要注意大批量操作数据时可能造成内存溢出,可以用evict,clear方法清除缓存中的内容;
//第一次查找,将此对象放入缓存中
Emp emp = (Emp) session.get(Emp.class, 3); System.out.println(emp.getEmpName());
//这个是从缓存中查找的,下面的get不会查找数据,从缓存中查找
Emp emp2 = (Emp) session.get(Emp.class, 3);
System.out.println(emp2.getEmpName());
二级缓存(sessionFactory级)
·配置:在hibernate.cfg.xml中配置cache.provider_class参数,hibernate内置了对Ehcache,OSCache,TreeCache,SwarmCache的支持
· 配置了总的开关还不行,还需要配置个<class-cache class="calssName" usage="read-only"/>来配置usage的值有read-write,nonstrict-read-write,transaction等
还可以通过每个类的配置文件中配置:ClassName.hbm.xml中:
在class的后面,id的前面配置<cache usage=""/>
·通过SessionFactory.evict(className.class) 和 evict(className.class, id )来清除设定的类的二级缓存
·session中的save(id的生成策略不能是native),saveOrUpdate,list, iterator ,get ,load 以及Query,Criteria都会填充二级缓存,session的iterator,get , load都会充二级缓存中读取数据(iterator可能存在N+1次查询)
--------------
session.save(dep); //这里不会放入缓存中,ID的生成策略为native,如果是saveOrUpdate的话,那么下面那个get方法就会出现缓存
Dep dep2 = (Dep) session.get(Dep.class, 3);
-------------
查询缓存:
-- 由于Query和Criteria的命中率比较低,hibernate默认是关闭的,对于hql的查询,Query,Criteria来说,要设置查询缓存要在hibernate.cfg.XML中配置 cache.use_query_cache 为true;
分布式缓存
中央缓存
·flush
Session.flush是用来同步缓存和数据库的,把缓存中的对象如果在数据库中没有的话,或者有更新的话,就把缓存中的东西更新到数据库中
·Id的生成策略如果是hilo,高地位生成策略:(会生成一个表来保存)对于这个生成策略,不要等到session.commit();提交之后才能生成id,他不需要进入数据库就可以生成id,所以如果是它的生成策略的话,save()方法不会直接进入数据库,而是会放入缓存中,
session.save(dep);
session.save(emp1);
session.save(emp2);
//session.clear(); 如果有这个语句的话,那么emp1,和emp2不会被保存
System.out.println("------");
transaction.commit(); //在这里提交之后才会生成sql语句,说明
·Id的生成策略如果是native的话,那么他的生成需要到数据中才能生成
批量增加的时候可以用到flush:
public void saveAll(List<Book> bookList){
final List<Book> finalBookList = bookList;
getHibernateTemplate().execute(new HibernateCallback(){
public Object doInHibernate(Session session)
throws HibernateException, SQLException {
int num = finalBookList.size();
for(int i = 0 ; i < num ; i++){
save(finalBookList.get(i));
if(i%20==0){
session.flush();
session.clear();
}
}
session.close();
return null;
}
});
}