zoukankan      html  css  js  c++  java
  • 不得不玩玩NHibernate

    1.0=>前言

    放着好好的EF不用,为什么要来玩NHibernate了?那是因为现在的工作内容就是维护一个比较老的项目,第一版是公司找外包做的,跟数据库打交道这块用的NHibernate,虽然都是ORM,但是既然维护这个项目有一段时间要一直搞,那还是趁着周末休息的时间先自己玩玩NHibernate,提前熟悉下.

    什么是NHibernate:

    1.开源的ORM(对象关系映射)框架.ORM可以被认为是,基于关系型数据库的数据存储,实现一个虚拟的面向对象的数据访问接口。理想情况下,基于这样一个面向对象的接口,持久化一个OO对象应该不需要要了解任何关系型数据库存储数据的实现细节。

    2.简化数据库操作的开发.功能 : 将 .Net实体对象 直接持久化 到数据库.

    在ADO.NET基础上进行封装,只需要少量代码就可以完成持久化工作,通过映射文件保存映射信息,在业务层以面向对象的方式编程,不用考虑数据保存形式

    2.0=>使用Hibernate的"3个准备,7个步骤"

    一.=> 3个准备 :
          a-添加程序集;
          b-配置文件;
          c-添加实体类和实体配置文件;
    添加程序集:你可以去https://sourceforge.net/projects/nhibernate/下载程序集也可以用Nuget,咱们这里直接用Nuget安装
    配置文件:无论你是自己下的还是用Nuget安装的都会有ConfigurationTemplates文件夹,找到MSSQL.cfg.xml复制一份作为你项目的NHibernate的配置文件,当项目使用很多框架的时候可能配置文件很多这个时候可以建一个配置文件的文件夹
    添加实体类和实体配置文件:本次Demo是用的动软代码生成器生成的,当然了你也可以用其他的生成器比如MyGeneration之类的,这个不是重点
    下面是本次小Demo的项目一览图(代码下载:NHibernate.rar数据库就采用NHibernate开源项目提供的示例数据库)
    二.=>7个步骤 : 
     
    当然了第一二步骤每次都写一次多么麻烦,所以这里我假如了一个NHibernateHelper,
     1  public class NHibernateHelper
     2     {
     3         private static ISessionFactory _sessionFactory;
     4         private static object _myLocker = new object();
     5         private static ISessionFactory SessionFactory
     6         {
     7             get
     8             {
     9                 if (_sessionFactory == null)
    10                 {
    11                     lock (_myLocker)
    12                     {
    13                         if (_sessionFactory == null)
    14                         {
    15                             _sessionFactory = new Configuration().Configure("NHibernateConfiguration/NHibernate.cfg.xml").BuildSessionFactory();
    16                         }
    17                     }
    18                 }
    19                 return _sessionFactory;
    20             }
    21         }
    22         public static ISession OpenSession()
    23         {
    24             try
    25             {
    26                 return SessionFactory.OpenSession();
    27             }
    28             catch (Exception ex)
    29             {
    30 
    31                 return null;
    32             }
    33 
    34         }
    35 
    36     }
    View Code
    SessionFactory是可以被共享使用的, 是线程安全的

    创建 会话(非共享使用,非线程安全,但是如果针对当前请求里的每个数据库操作都创建 会话,则极度浪费资源,所以,最好是 针对每次请求 只创建一个 会话,后面从当前上下文里面直接取,本次demo还是每次都直接开启一个了)
    ISession sess = SessionFactory.OpenSession();
    HttpContext.Current.Items.Add("nfsession", sess);

    3.0=>ISession的几个主要方法

    1.Save ,Persist保存数据. 后者在事务外不会产生insert语句.
    2.Delete删除对象
    3.Update更新对象,如果数据库中没有记录,会出现异常
    4.Get,根据id立即查询数据库
    5.Load 根据id查(返回的是代理,不回立即访问数据库)
    6.SaveOrUpdate,Merge(根据id和version的值来确定是Save或Update),调用Merge你的对象还是托管的.
    7.Lock,把对象变成持久独享,但不会同步对象的状态.
     
    这里强调下SaveOrUpdate这个方法,当把model传给方法的时候如果model指定主键id那么会去判断如果数据库有这个id对应的数据记录就会做Update操作,如果model没有指定主键id就做Save新增操作,当你不清楚你即将做什么操作的时候可以用这个方法,一般还是用Save和Update就足够.
    另外说一下用ISession的时候可以用事务配合使用和可以不用事务,用事务的时候如果做的是批量操作有错误会回滚,反之不用事务没有回滚,如果你的数据操作逻辑比较多,就可以使用事务.
     1  //1.创建工厂
     2             using (ISessionFactory fac = new Configuration().Configure().BuildSessionFactory())
     3             {
     4                 //2.创建会话
     5                 using (ISession sess = fac.OpenSession())
     6                 {
     7                     //3.开启事务 (此时,sess对象已经被传入到tran对象中了,所以当commit的时候,就可以 调用到此sess的Flush方法)
     8                     using (ITransaction tran = sess.BeginTransaction())
     9                     {
    10                         //构造要删除的 对象
    11                         Classes model = new Classes() { CID = int.Parse(strId) };
    12                         //删除
    13                         sess.Delete(model);
    14                         //提交事务
    15                         tran.Commit();
    16                         context.Response.Write("删除成功!");
    17                     }
    18                 }
    19             }

    开启事务 (此时,sess对象已经被传入到tran对象中了,所以当commit的时候,就可以 调用到此sess的Flush方法)

    4.0=>HQL和Criteria

    HQL(Hibernate Query Language)
    面向对象的查询语言,与SQL不同,HQL中的对象名是区分大小写的(除了C#类和属性其他部分不区分大小写);
    HQL中查的是对象而不是表,并且支持多态;
    HQL主要通过Query来操作,Query的创建方式:
    IQuery q = sess.CreateQuery(hql);
    from Person
    from User u where u.name=:name
    from User u where u.name=:name and u.birthday<:birthday
    l
    Criteria
    Criteria 是一种比HQL更面向对象的查询方式;
    ICriteria crit = sess.CreateCriteria(typeof(User));
    简单属性条件如:crit.Add(Restrictions.eq(proName,value));
    crit.Add(Restrictions.eqProperty(proName,otherProName));

    5.0=>NHibernate缓存

    NHibernate一级缓存

    即ISession缓存,属于事务级缓存,只能在ISession生命周期内使用.
    1.每次查询数据结果会存储在此Isession中,关闭前做相同查询就直接获取,而不查数据库.
    2.此缓存存在于各个ISession中,不同ISession对象的缓存不共享.
    一级缓存操作方法:
    1. ISession.Evict(object):删除指定实例
    2. ISession.Clear():清空
    3. ISession.Contains(object):检查是否包含指定实例
    NHibernate二级缓存
    由ISessionFactory维护,被所有ISession共享.
    数据检索方式:先去一级缓存找,再到二级缓存找,如果都没有,才去数据库查询.
    缓存使用原则:经常被访问,但修改次数少.

    最后附上一个帮助文档:NHIBERNATE-符合.Net习惯的关系数据库持久化

    文档里面很全面,基本看着帮助文档就可以自己玩起来了
  • 相关阅读:
    CSS 页面锚点设置
    CSS尺寸 物理像素、CSS像素、dip、dpr、ppi、dpi(一)
    项目代码格式化规范
    Vue 计算属性与监听属性区别
    Python操作InfluxDB
    InfluxDB学习
    Redis学习(五)| 哈希 Hash
    Redis学习(四)| 字符串 String
    Redis学习(三)| 命令、键
    Redis学习(二)| 数据类型
  • 原文地址:https://www.cnblogs.com/zhangyihui/p/5403305.html
Copyright © 2011-2022 走看看