摘要

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对象二级缓存上的更多相关文章

  1. NHibernate系列文章目录

    第一章:NHibernate基础 NHibernate介绍 第一个NHibernate工程 简单的增删改查询 运行时监控 NHibernate配置 数据类型映射 Get/Load方法 NHiberna ...

  2. NHibernate系列文章十:NHibernate对象二级缓存下

    摘要 上一节对NHibernate二级缓存做了简单介绍,NHibernate二级缓存是由SessionFactory管理的,所有Session共享.这一节介绍二级缓存其他两个方面:二级缓存查询和二级缓 ...

  3. NHibernate系列文章八:NHibernate对象一级缓存

    摘要 Nhibernatea缓存非常强大,按照缓存存储在Session对象还是SessionFactory对象分为一级缓存和二级缓存. 一级缓存存在于Session对象里,也叫Session缓存,由S ...

  4. NHibernate系列文章七:NHibernate对象状态

    摘要 NHibernate对象持久化 NHibernate对象的三个状态:临时态.持久态.游离态(托管态) NHibernate三状态的相互转化 1. NHibernate对象持久化 NHiberna ...

  5. NHibernate系列文章一:NHibernate介绍

    摘要 NHibernate是一个成熟的开源的面向对象的.net映射框架.大量的实际项目中正在使用该框架.他是建立在ADO.Net基础之上.目前的版本是NHibernate 4.0.4.本系列文章都是基 ...

  6. NHibernate系列文章十七:NHibernate Session管理(附程序下载)

    摘要 NHibernate的Session的管理涉及到NHibernate的两个最重要的对象ISessionFactory和ISession.ISessionFactory的生成非常消耗资源,通常都在 ...

  7. NHibernate系列文章二十七:NHibernate Mapping之Fluent Mapping基础(附程序下载)

    摘要 从这一节起,介绍NHibernate Mapping的内容.前面文章都是使用的NHibernate XML Mapping.NHibernate XML Mapping是NHibernate最早 ...

  8. NHibernate系列文章二十三:NHibernate查询之Criteria查询(附程序下载)

    摘要 上一篇文章介绍了NHibernate HQL,他的缺点是不能够在编译时发现问题.如果数据库表结构有改动引起了实体关系映射的类有改动,要同时修改这些HQL字符串.这篇文章介绍NHibernate面 ...

  9. NHibernate系列文章十一:NHibernate并发控制

    摘要 在同一时刻数据访问量和更新次数比较大的系统中,产生了数据的并发访问问题.并发访问使得在这样的环境中,所有用户(程序.实际用户.进程.线程等)的操作不产生负面问题. 如果不使用并发,在两个用户同时 ...

随机推荐

  1. 安装生物信息学软件-Biopython

    其实好多东西装过好多次,然而每次都要翻文档,经常掉进前面掉进过的坑...所以这里重新写一份指南,以防下次再装又忘了(魂淡我并不想再装了啊不要立flag) 1. 安装biopython 1.1 因为bi ...

  2. C++虚函数的实现机制示例

    C++虚函数的实现机制是通过一个vtable表,指向子类的虚函数地址. 另外,如果不是虚函数,则不能实现用父类引用调用子类方法. #include <windows.h> #include ...

  3. Android使用SharedPreference存储数据

    SharedPreference存储数据和文件存储更加方便的一点是可以按照一定的数据类型进行存储,同时取数据时也能够获取到相应的数据类型.它是按照map的方式来存储和读取数据的. MainActivi ...

  4. 数据结构《16》----自动补齐实现《一》----Trie 树

    1. 简述 Trie 树是一种高效的字符串查找的数据结构.可用于搜索引擎中词频统计,自动补齐等. 在一个Trie 树中插入.查找某个单词的时间复杂度是 O(len), len是单词的长度. 如果采用平 ...

  5. HDU5898、 HDU 2089(数位DP)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5898 题意:很明确,找出区间[l , r]中符合连续奇数为偶数,连续偶数为奇数的个数. 思路:dp[i ...

  6. springmvc学习笔记--Interceptor机制和实践

    前言: Spring的AOP理念, 以及j2ee中责任链(过滤器链)的设计模式, 确实深入人心, 处处可以看到它的身影. 这次借项目空闲, 来总结一下SpringMVC的Interceptor机制, ...

  7. dedecms搜索框制作

    <form method=" name="kwtype"> <table width="> <tr> <td widt ...

  8. 理解伪元素 :before和:after

    在CSS中有一特性允许我们添加额外元素而不扰乱文档本身,就是伪元素. 下面举例说明: html: <blockquote>  blockquote只是一个块引用 </blockquo ...

  9. Python error: ascii’/'utf-8′ codec can’t decode byte 0xb8 in position 50: ord

    字符串使用了费ascii编码的字符,也就是它代表的16进制的编码超过127. 解决这个问题可以使用下面的方法解决,其实就是设置默认的编码.python 2.x的默认编码是ascii,如果改为utf-8 ...

  10. [Spring] IOC - Annotation

    Spring Annotation使用例子. 与XML配置的例子一样:http://www.cnblogs.com/HD/p/3962541.html Project结构: 配置文件:springCo ...