NHibernate系列文章九:NHibernate对象二级缓存上
摘要
NHibernate的二级缓存由SessionFactory管理,由所有Session共享。
NHibernate缓存读取顺序:
首先从一级缓存中读取,如果一级缓存对象存在,则读取一级缓存对象并返回
- 如果一级缓存对象不存在,则读二级缓存。
- 如果二级缓存对象存在,则读取二级缓存里的对象并返回
- 如果二级缓存对象不存在,则读数据库
二级缓存一般存储不发生变化的数据,如果数据经常要进行修改删除操作,不宜使用缓存
1. NHibernate二级缓存设置
NHibernate的二级缓存在hibernate.cfg.xml文件中设置。
- cache.provider_class:二级缓存提供程序。NHibernate自带二级缓存提供程序类NHibernate.Cache.HashtableCacheProvider(许多书上说实际项目中建议不要使用这个类,这里只做为示例使用它)。如果需要使用第三方NHibernate二级缓存提供类,则需要在程序里添加第三方dll引用,并在cache.provider_class指向第三方类。这些第三方缓存provider都是开源的,可以从NHibernate Contrib下载。
- cache.use_second_level_cache:设置是否使用二级缓存
- class-cache:为NHibernate实体类设置缓存策略,格式是:
<class-cache class="完整类名,Assembly名称" usage="read-only|read-write|nonstrict-read-write|transactional"/>
class是NHibernate实体类,以逗号隔开,前部分是完整的类名,后面部分是assembly名称。
usage是并发访问策略,NHibernate内置四种并发访问策略,可以设置下面四个值:
- read-only:只读缓存,适用于只读数据。
- read-write:读写缓存。
- nonstrict-read-write:非严格读写缓存,不保证缓存与数据库的一致性。
- transactional:事务缓存,提供可重复读的事务隔离级别。
一般只需要设置成read-only或者read-write,这篇文章第二部分介绍更新操作对二级缓存的影响。
- 除了在hibernate.cfg.xml文件中设置class-cache指定为哪一个实体类设置二级缓存以外,也可以在实体关系映射配置文件内设置。
设置方法是在id属性之前添加这句:<cache usage="read-only|read-write|nonstrict-read-write|transactional" />
- 推荐统一在hibernate.cfg.xml文件中设置,后期维护更方便。
程序演示
修改hibernate.cfg.xml文件
<?xml version="1.0" encoding="utf-8" ?>
<hibernate-configuration xmlns="urn:nhibernate-configuration-2.2">
<session-factory>
<property name="connection.connection_string_name">default</property>
<property name="connection.driver_class">NHibernate.Driver.SqlClientDriver</property>
<property name="dialect">NHibernate.Dialect.MsSql2008Dialect</property>
<property name="cache.provider_class">NHibernate.Cache.HashtableCacheProvider</property>
<property name="cache.use_second_level_cache">true</property>
<mapping assembly="NHibernateDemoApp"/>
<class-cache class="NHibernateDemoApp.Customer,NHibernateDemoApp" usage="read-write"/>
</session-factory>
</hibernate-configuration>
修改Main函数
static void Main(string[] args)
{
HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize(); //在两个session里两次查询相同对象
using (var session = SessionFactory.OpenSession())
{
var customer = session.Get<Customer>();
} using (var session = SessionFactory.OpenSession())
{
var customer = session.Get<Customer>();
} Console.WriteLine("Completed");
Console.ReadLine();
}
打开NHibernateProfile,ctrl+x清空所有Session,F5执行应用程序得到结果

结果显示只查询了一次数据库。因为是两独立session两次查询同一个持久化对象,如果没有使用二级缓存,应该是两次查询数据库,结果跟上一篇文章一样。但是这里使用了二级缓存,二级缓存在所有ISession里共享,因此只查询一次数据库。
NHibernateProfile的结果第二行内容,也明确表明了,第二次Get的时候是从二级缓存中读取的。
2. 更新操作对二级缓存的影响
更新操作对二级缓存的影响由配置项class-cache的属性usage决定,如果设置的是read-only,则数据更新后不刷新缓存。如果设置的是read-write,则数据更新后刷新二级缓存,第二次Session查询将重新查询数据库。
程序演示
修改Main函数
static void Main(string[] args)
{
HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize(); using (var session = SessionFactory.OpenSession())
{
Console.WriteLine("in session 1");
var customer = session.Get<Customer>();
Console.WriteLine("before updated: {0} {1}", customer.FirstName, customer.LastName);
customer.LastName = "Chen";
session.SaveOrUpdate(customer);
session.Flush();
Console.WriteLine("after updated: {0} {1}", customer.FirstName, customer.LastName);
} using (var session = SessionFactory.OpenSession())
{
Console.WriteLine("in session 2");
var customer = session.Get<Customer>();
Console.WriteLine("{0} {1}", customer.FirstName, customer.LastName);
} Console.WriteLine("Completed");
Console.ReadLine();
}
ctrl+x清空NHibernateProfile的Session,F5执行程序,得到结果

LastName已经得到更新,第二次session查询得到了最新的记录。

第二次session查询自动查询数据库。
NHibernate系列文章九:NHibernate对象二级缓存上的更多相关文章
- NHibernate系列文章目录
第一章:NHibernate基础 NHibernate介绍 第一个NHibernate工程 简单的增删改查询 运行时监控 NHibernate配置 数据类型映射 Get/Load方法 NHiberna ...
- NHibernate系列文章十:NHibernate对象二级缓存下
摘要 上一节对NHibernate二级缓存做了简单介绍,NHibernate二级缓存是由SessionFactory管理的,所有Session共享.这一节介绍二级缓存其他两个方面:二级缓存查询和二级缓 ...
- NHibernate系列文章八:NHibernate对象一级缓存
摘要 Nhibernatea缓存非常强大,按照缓存存储在Session对象还是SessionFactory对象分为一级缓存和二级缓存. 一级缓存存在于Session对象里,也叫Session缓存,由S ...
- NHibernate系列文章七:NHibernate对象状态
摘要 NHibernate对象持久化 NHibernate对象的三个状态:临时态.持久态.游离态(托管态) NHibernate三状态的相互转化 1. NHibernate对象持久化 NHiberna ...
- NHibernate系列文章一:NHibernate介绍
摘要 NHibernate是一个成熟的开源的面向对象的.net映射框架.大量的实际项目中正在使用该框架.他是建立在ADO.Net基础之上.目前的版本是NHibernate 4.0.4.本系列文章都是基 ...
- NHibernate系列文章十七:NHibernate Session管理(附程序下载)
摘要 NHibernate的Session的管理涉及到NHibernate的两个最重要的对象ISessionFactory和ISession.ISessionFactory的生成非常消耗资源,通常都在 ...
- NHibernate系列文章二十七:NHibernate Mapping之Fluent Mapping基础(附程序下载)
摘要 从这一节起,介绍NHibernate Mapping的内容.前面文章都是使用的NHibernate XML Mapping.NHibernate XML Mapping是NHibernate最早 ...
- NHibernate系列文章二十三:NHibernate查询之Criteria查询(附程序下载)
摘要 上一篇文章介绍了NHibernate HQL,他的缺点是不能够在编译时发现问题.如果数据库表结构有改动引起了实体关系映射的类有改动,要同时修改这些HQL字符串.这篇文章介绍NHibernate面 ...
- NHibernate系列文章十一:NHibernate并发控制
摘要 在同一时刻数据访问量和更新次数比较大的系统中,产生了数据的并发访问问题.并发访问使得在这样的环境中,所有用户(程序.实际用户.进程.线程等)的操作不产生负面问题. 如果不使用并发,在两个用户同时 ...
随机推荐
- cookie窃取和session劫持
Updates 2014-08-17 感谢@搞前端的crosser的提醒,加入了HTTP Response Splitting的内容. 此篇文章的Presentation戳这里. 一.cookie的基 ...
- How to implement updatable view with NHibernate
see : http://msdn.microsoft.com/en-us/library/ms187956.aspx(The constrains of creation updatable vie ...
- UVALive 7338 (树链剖分+线段树)
Problem Toll Management IV 题目大意 给一张n个点m条边的无向图,有边权.数据保证前n-1条边构成了一棵最小生成树. 要求对于每条边求出其边权上下最多浮动范围,使得最小生成树 ...
- JSP 动作元素
JSP动作元素 1. 动作元素分类 用来动态的包含文件.网页跳转及使用JavaBean组件等. 语法:<jsp:XXX />或者<jsp:XXX></jsp:XXX&g ...
- linux下安装mongodb(php版本5.3)
转:原文出处忘记了. 1. 下载MongoDB 2. 解压文件到某目录下,然后重命名: [root@localhost var]# tar -xzvf mongodb-linux-i686-2.0.1 ...
- 循环神经网络(RNN, Recurrent Neural Networks)介绍(转载)
循环神经网络(RNN, Recurrent Neural Networks)介绍 这篇文章很多内容是参考:http://www.wildml.com/2015/09/recurrent-neur ...
- 深入解读Linux与Android的相互关系(转-lining)
大家都知道Android是基于Linux内核的操作系统,也曾经和Linux基金会因为内核问题产生过分歧,本文将开始对Android的内核进行剖析,主要介绍Android和Linux之间的关系,后续还会 ...
- ES pom配置
https://github.com/elastic/elasticsearch/issues/19415 <dependency> <groupId>org.elastics ...
- maxscript, 批量导出物体
1,将场景中所有选中物体整体导出为一个fbx文件 exportfile filename #noprompt selectedOnly:true using:FBXEXP 2,将场景中所有选中物体各导 ...
- SQLServer2012中用于记录数据操作时刻的附加字段使用datetime2(3)就可以了
datetime2(3)精确到毫秒(听说),约等于2005时代的datetime类型.实际上后者是精确到3.33毫秒(也是听说). ) = GETDATE(); ) = GETDATE(); ) = ...