NHibernate的一级缓存,名词好像很牛B,很难。实际上就是ISession缓存。存储在ISession的运行周期内。而二级缓存则存储在ISessionFactory内。

一、ISession一级缓存测试

  ISession默认开启一级缓存,不需要任何配置。ISession缓存在ISession创建后就可以使用,以后每次通过此ISession操作数据时,ISession会检测自身是否缓存有相应的数据,如果有则直接返回。如果没有,这时才查询数据库返回,同时缓存到ISession。当释放ISession实例的时候,缓存自动销毁。

  示例:

  首先我们先开启配置文件中的show_sql节点:

  <property name="show_sql">true</property>  //开启这个对本缓存测试没影响,只是看看效果

  PersonDao.cs

    public class PersonDao
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
ISession session; public PersonDao()
{
session = sessionFactory.OpenSession();
}
public PersonModel GetPersonById(int Id)
{
PersonModel p = session.Get<PersonModel>(Id);
return p;
}
}

  Program.cs

    class Program
{
static void Main(string[] args)
{
PersonDao dao = new PersonDao();
//第一次读取
PersonModel p1 = dao.GetPersonById();
Console.WriteLine(p1.Id + " " + p1.Name);
//第二次读取
PersonModel p2 = dao.GetPersonById();
Console.WriteLine(p2.Id + " " + p2.Name); Console.ReadKey();
}
}

  输出:

  

  我们看到只有第一次查询执行了SQL语句,第二次并没有执行任何语句。这点从SQL Server Profiler中也可监控到。

  

  在两个会话中获得同一实例

  下面我们把

    sessionFactory.OpenSession();

  这行代码放到方法里面会怎么样呢?

        public PersonModel GetPersonById(int Id)
{
ISession session = sessionFactory.OpenSession();
PersonModel p = session.Get<PersonModel>(Id);
return p;
}

  输出结果如下:

  

  留意到每OpenSession一次,缓存就没有了。OpenSession一次,NHibernate会自动释放原有的ISession对象。

  看到这里,你应该懂得注意下自己代码的写法了。

二、ISession.Get()与ISession.Load()的区别

  不想废话,十几行代码说明问题:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
ISession session = sessionFactory.OpenSession(); Console.WriteLine("Get之前");
PersonModel pGet = session.Get<PersonModel>();
Console.WriteLine("Get之后");
Console.WriteLine("读取名字" + pGet.Name);
Console.WriteLine("=====================================");
Console.WriteLine("Load之前");
PersonModel pLoad = session.Load<PersonModel>();
Console.WriteLine("Load之后");
Console.WriteLine("读取名字" + pLoad.Name); Console.ReadKey();
}

  输出结果如下:

  

  留意到Get()执行的时候,就立即到数据库读取了。而Load()是延迟加载,到要用的时候,采取数据库读取,不用不读。

  不用Get<>或Load<>能否缓存

  下面我们把PersonDao改成这样:

    public class PersonDao
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
ISession session; public PersonDao()
{
session = sessionFactory.OpenSession();
} public IList<PersonModel> GetPersonList()
{
return session.QueryOver<PersonModel>().List();
}
}

  Program.cs

        static void Main(string[] args)
{
PersonDao dao = new PersonDao();
IList<PersonModel> PersonList1 = dao.GetPersonList();
Console.WriteLine(PersonList1[].Name); IList<PersonModel> PersonList2 = dao.GetPersonList();
Console.WriteLine(PersonList1[].Name); Console.ReadKey();
}

  输出:

  

  留意到,还是执行了两次SQL语句。看来,只有Get()或Load()才能使用NHibernate的一级缓存。

三、一级缓存的管理

  当我们使用Get()或Load()的时候,不想使用一级缓存怎么办呢?答案,使用

  如我们回到本文的第一个示例,只改如下加粗的代码:

    public class PersonDao
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
IStatelessSession session; public PersonDao()
{
session = sessionFactory.OpenStatelessSession();
} public PersonModel GetPersonById(int Id)
{
PersonModel p = session.Get<PersonModel>(Id);
return p;
}
}

  Program.cs

        static void Main(string[] args)
{
PersonDao dao = new PersonDao();
//第一次读取
PersonModel p1 = dao.GetPersonById();
Console.WriteLine(p1.Id + " " + p1.Name);
//第二次读取
PersonModel p2 = dao.GetPersonById();
Console.WriteLine(p2.Id + " " + p2.Name); Console.ReadKey();
}

  输出结果如下:

  

  可以看到NHibernate还是老老实实执行了两次SQL语句。

  NHibernate提供了如下三个方法,让我们管理一级缓存。

  • ISession.Evict(object):从缓存中删除指定实例。
  • ISession.Clear():清空缓存。
  • ISession.Contains(object):检查缓存中是否包含指定实例。
  • ISession.Refresh(object):刷新缓存中的单个实例。

  一个示例说明问题:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
ISession session = sessionFactory.OpenSession(); PersonModel p1 = session.Get<PersonModel>();
PersonModel p2 = session.Get<PersonModel>();
PersonModel p3 = new PersonModel();
Console.WriteLine(session.Contains(p1)); //输出True
Console.WriteLine(session.Contains(p2)); //输出True
session.Evict(p1); //从一级缓存中移除p1,看你还Con不Contains
Console.WriteLine(session.Contains(p1)); //输出False
Console.WriteLine(session.Contains(p2)); //输出True
session.Clear(); //全部清空,看你还Con不Containsp2
Console.WriteLine(session.Contains(p2)); //输出False Console.ReadKey();
}

  输出结果如下:

  

  刷新缓存:

        static void Main(string[] args)
{
ISessionFactory sessionFactory = new Configuration().Configure().BuildSessionFactory();
ISession session = sessionFactory.OpenSession(); PersonModel p1 = session.Get<PersonModel>();
Thread.Sleep();
Console.WriteLine("3秒钟之后----------------");
session.Refresh(p1); Console.ReadKey();
}

  输出结果:

  

  可以看到3秒钟之后,NHibernate重新查询了一次数据库,获得最新的对象。

NHibernate之一级缓存(第十篇)的更多相关文章

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

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

  2. [Nhibernate]一级缓存

    目录 写在前面 文档与系列文章 一级缓存 一个例子 一级缓存管理 总结 写在前面 上篇文章介绍了nhibernate中对象的三种状态,通过对象的三种状态,很容易想到缓存. 什麽是缓存? 有时候,某些数 ...

  3. NHibernate教程(19) —— 一级缓存

    本节内容 引入 NHibernate一级缓存介绍 NHibernate一级缓存管理 结语 引入 大家看看上一篇了吗?对象状态.这很容易延伸到NHibernate的缓存.在项目中我们灵活的使用NHibe ...

  4. NHibernate二级缓存(第十一篇)

    NHibernate二级缓存(第十一篇) 一.NHibernate二级缓存简介 NHibernate由ISessionFactory创建,可以被所有的ISession共享. 注意NHibernate查 ...

  5. 01-08-01【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate中的一级缓存

    缓存的范围? 1.事务范围 事务范围的缓存只能被当前事务访问,每个事务都有各自的缓存,缓存内的数据通常采用相互关联的对象形式.缓存的生命周期依赖于事务的生命周期,只有当事务结束时,缓存的生命周期才会结 ...

  6. Python开发【第二十篇】:缓存

    Python开发[第二十篇]:缓存redis&Memcache   点击这里 Python之路[第九篇]:Python操作 RabbitMQ.Redis.Memcache.SQLAlchemy ...

  7. Mybatis第八篇【一级缓存、二级缓存、与ehcache整合】

    Mybatis缓存 缓存的意义 将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题. myba ...

  8. Hibernate第七篇【对象状态、一级缓存】

    前言 本博文主要讲解Hibernate的细节-->对象的状态和一级缓存- 对象状态 Hibernate中对象的状态: - 临时/瞬时状态 - 持久化状态 - 游离状态 学习Hibernate的对 ...

  9. 【串线篇】Mybatis缓存之一级缓存

    1.体会 一级缓存:MyBatis:SqlSesion级别的缓存:默认存在,不需要设置. 机制:只要之前查询过的数据,mybatis就会保存在一个缓存中(Map):下次获取直接从缓存中拿:当前sess ...

随机推荐

  1. PHP 接入(第三方登录)QQ 登录 OAuth2.0 过程中遇到的坑

    前言 绝大多数网站都集成了第三方登录,降低了注册门槛,增强了用户体验.最近看了看 QQ 互联上 QQ 登录的接口文档.接入 QQ 登录的一般流程是这样的:先申请开发者 -> 然后创建应用(拿到一 ...

  2. 自动化测试===uiautomator2类似appium

    项目地址:https://github.com/openatx/uiautomator2 http://mp.weixin.qq.com/s/YKxwW-pL603Fll3QIWapVw https: ...

  3. C#反射动态调用dll中的方法及使用QuartZ.net实现作业调度

    using Quartz; using Quartz.Impl; using System; using System.Collections.Generic; using System.Linq; ...

  4. C基础入门 - 第一章 - C语言绪言

    第1章 C语言绪言 1.1 C语言概述 1.1.1 C语言世界 1.1.2 C语言学习, 能当饭吃吗 1.2 开发环境构建 1.2.1 visual studio安装使用 1.2.2 visual s ...

  5. MYSQL表中设置字段类型为TIMESTAMP时的注意事项

    在MYSQL中,TIMESTAMP类型是用来表示日期的,但是和DATETIME不同,不同点就不再这里说明了. 当我们在使用TIMESTAMP类型设置表中的字段时,我们应该要注意一点,首先我们在表中新增 ...

  6. leetcode 之Set Matrix Zeroes(10)

    设置两个布尔数组,记录行和列是否存在0.需要注意的是如何将行或列设为0. void setZeros(vector<vector<int>> &matrix) { in ...

  7. 微信小程序获取用户信息“授权失败”场景的处理

    很多的时候我们在处理小程序功能的时候需要用户获取用户信息,但是呢为了信息安全,用户不授权导致授权失败场景:但是小程序第二次不在启动授权信息弹层,为了用户体验,可以用以下方式处理: function i ...

  8. 【JBPM4】判断节点decision 方法2 condition

    运用condition计算并判断. JPDL <?xml version="1.0" encoding="UTF-8"?> <process ...

  9. 这个程序员有点牛,现场直接用JS写了个飞机游戏,半小时吸粉三千

    程序员昨晚在b站直播的时用JavaScript代码写了一个飞机大战游戏,半小时不到粉丝关注就上千了. 今日就拿出来跟大家分享一下,对许多大佬来说做这个特效也不是很难,但是对于刚开始学习前端这方面还是有 ...

  10. bzoj 1443 二分图博弈

    这种两个人轮流走,不能走 走过的格子的大都是二分图博弈... #include<bits/stdc++.h> #define LL long long #define fi first # ...