首先是配置文件中配置正确,META-INF中加上二级缓存的配置:
<property name="hibernate.cache.provider_class" value="org.hibernate.cache.EhCacheProvider" /><property name="hibernate.generate_statistics" value="true" />
<property name="hibernate.cache.use_second_level_cache"value="true" /><property name="hibernate.cache.use_query_cache" value="true"
注意如果你的项目开始的时候可能没使用二级缓存,那个时候可能会有一行配置文件:
<property name="hibernate.cache.provider_class" value="org.hibernate.cache.NoCacheProvider" />
千万要注意看配置文件中是否有上述一行的配置,否则的话就会一直报如下的错误。主要原因就是,上述配置大体意思是一个无cache的配置,但是你当前配置的其他部分配置了要使用cache,导致了初始化文件不知道如何进行初始化,引发错误。
Caused by: org.hibernate.cache.NoCachingEnabledException: Second-level cache is not enabled for usage [hibernate.cache.use_second_level_cache | hibernate.cache.use_query_cache]
at org.hibernate.cache.NoCacheProvider.buildCache(NoCacheProvider.java:21)
at org.hibernate.cache.CacheFactory.createCache(CacheFactory.java:61)
... 84 more
配置文件配置好,就是jpa的entity类的配置了
import org.hibernate.annotations.Cache;
import org.hibernate.annotations.CacheConcurrencyStrategy;
@Entity
@Table(name = "lists", catalog = "release")
@Cache(usage = CacheConcurrencyStrategy.READ_ONLY)
public class Lists implements java.io.Serializable {
上面import需要注意,因为可能有其他包里会有cache类的注解。其中CacheConcurrencyStrategy有几种方式,随便一搜就可以看各个使用场景。其中read_only只能是新建之后就不能再对其更新了。
另外查询的时候需要注意:
public Lists find(Integer id){
Query query = getEntityManager().createQuery("select model from " + getClassName() + " model where model.id = " + id);
if(query instanceof org.hibernate.ejb.QueryImpl){
((org.hibernate.ejb.QueryImpl) query).getHibernateQuery().setCacheable(true);
return (Lists) query.getSingleResult();
}
return null;
}
需要把jpa类型的Query转化为Hibernate类型的query,因为jpa类型的query没有设置setCacheable(true)的方法,就无法使用二级缓存。所以必须显示的类型转化成hibernate的query一次才可以。
关于为何可以转化,是因为Hibernate的query其实也是实现了javax.persistence.Query的接口,所以可以转化。
对是否使用二级缓存的验证可以在配置文件中加入一行配置:
<property name="hibernate.show_sql" value="true" />
然后查看是否每次查询会用数据库语句的输出。