NHibernate数据库映射工具

NHibernate是一个面向.NET环境的对象/关系数据库映射工具。对象/关系数据库映射(object/relational mapping,ORM)这个术语表示一种技术,用来把对象模型表示的对象映射到基于SQL的关系模型数据结构中去。

21a4462309f79052fe98a5340cf3d7ca7bcbd531.jpg
NHibernate 是一个基于.Net 的针对关系型数据库的对象持久化类库。Nhibernate 来源于非常优秀的基于Java的Hibernate 关系型持久化工具。
NHibernate 从数据库底层来持久化你的.Net 对象到关系型数据库。NHibernate 为你处理这些,远胜于你不得不写SQL去从数据库存取对象。你的代码仅仅和对象关联,NHibernat 自动产生SQL语句,并确保对象提交到正确的表和字段中去。
官网下载:http://nhibernate.info/
NHibernate官方教程:http://nhibernate.info/doc/tutorials/first-nh-app/your-first-nhibernate-based-application.html(基于Server SQL的连接教程,但是原理一样)
Code Project里面的教程示例:https://www.codeproject.com/articles/26123/nhibernate-and-mysql-a-simple-example
Paste_Image.png

这里推荐一个网站Sourceforge:https://sourceforge.net/(很多开源插件都放在这个网站)
下载后直接在Required_Bins文件下找到NHibernate.dll将其添加入VS里面的引用就OK了

配置NHibernate

在这里我先摆上自己创建的数据库,然后开始配置映射

Paste_Image.png

我们现在必须告诉NHibernate我们想要使用哪个数据库产品,并以连接字符串的形式提供连接细节。NHibernate支持许多数据库产品!我们使用的是MySQL,所以我们先添加一个XML文本,并将其命名为hibernate.cfg.xml(固定命名)然后再xml文件下添加以下配置。

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-configuration xmlns="urn:nhibernate-configuration-2.2">
  <session-factory>
    <property name="connection.provider">NHibernate.Connection.DriverConnectionProvider</property>
    <property name="dialect">NHibernate.Dialect.MySQL5Dialect</property>
    <!--用来配置使用什么版本的数据库-->
    <property name="connection.driver_class">NHibernate.Driver.MySqlDataDriver</property>
    <!--用来配置使用什么数据库-->
    <property name="connection.connection_string">Server=localhost;Database=mygamedb;User ID=root;Password=root;</property>
    <!--配置MySQL的IP地址数据库名用户名和密码-->

    <property name="show_sql">true</property>
  </session-factory>
</hibernate-configuration>

使用这个配置文件,我们告诉NHibernate我们要使用MySQL作为目标数据库
connection.provider: 设置NHibernate要连接到数据库的连接提供程序。
dialect :陈述了NHibernate类名,可以实现某些依赖平台的功能,在这种情况下,由于我使用MySQL 5,所以明显的选择是MySQL5Dialect
connection.driver_class: 设置哪个驱动程序应该使用,在这种情况下,当使用MySQL时,MySqlDataDriver是一个合乎逻辑的选择
connection.connection_string: 是数据库的连接字符串。
配置文件的映射部分告诉NHibernate哪些映射文件用于对象/关系映射
配置完后,我们将其属性“ 复制到输出目录 ”设置为“ 始终复制 ”。

Paste_Image.png

配置数据的映射

配置对应的数据库后我们开始创建一个模型的类,首先要对应数据库中的数据的话我们需要创建一个类去保存数据库中的数据。这里我们先在项目下先创建出一个文件夹命名为Model,然后在Model文件夹下创建一个类,为了与数据库中对应的User表相对应我们在这里给类命名为User,然后写入和数据库中相对应的表中的属性,如下

 public   class User
    {
       //对应数据库的属性
        public virtual int Id { get; set; }
        public virtual string  Username { get; set; }
        public virtual string Password { get; set; }
        public virtual DateTime Registerdate { get; set; }
    }

为了能够使用User该类,有必要创建一个映射文件,其中包含NHibernate用于对象/关系映射的元数据,即连接类声明,属性到列和数据库表中的键。然后这个时候我们还需要一个配置文件,用来映射数据库中的属性到类库中,我们先创建一个Mappings文件夹,然后再Mappings文件夹下创建XML文件并且命名为User.hbm.xml,然后写入以下配置信息

NHibernate/Database(数据库)/.NET里面的各种类型的声明 :http://nhibernate.info/doc/nhibernate-reference/mapping.html#mapping-types

Paste_Image.png

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
                   assembly="NHibernateController"
                   namespace="NHibernateController.Model">
  <!--assembly:模型所在的程序集-->
  <!--namespace:表示User这个类在那个命名空间下的那个文件内-->
 <class name="User" table="user"><!--表示User这个类跟数据库里面的user表对应-->
    <id name="Id" column="id" type="Int32"><!--id name配置User类里面主键的映射,column表示跟数据库中哪个主键对应,type类型-->
      <generator class="native"> </generator><!--generator表示一个生成器,class指定generator的类型,native表示使用数据库自带的生成器-->
    </id>
        <property name="Username" column="UserName" type="String"> </property>
          <property name="Password" column="PassWord" type="String"> </property>
            <property name="Registerdate" column="registerdate" type="Date"></property>
  </class>
</hibernate-mapping>

映射文件包含一组节点,其简短版本的解释是:
hibernate-mapping:声明这是一个hibernate映射文件,该xmlns属性声明应该使用的XML命名空间

class: 指出此映射连接到的持久类:
   该name属性必须指定完全限定的.Net类名称,并且还必须包含程序集名称
   该table属性命名数据库中的表名
id: 是描述数据库表中主键列的节点
   该name属性告诉NHibernate永久类中的哪个属性被使用
   该column属性告诉数据库表中的哪个列是主键
   该type属性告诉NHibernate数据库类型,这在大多数情况下应该被自动检索,但是我发现这是一个问题,因此指定类型

generator:是id节点的必需子元素,该class属性声明应该使用哪个.NET类来为持久化类的实例生成唯一标识符,该类可能是应用程序的特定实现,也可以是其中一个NHibernate提供的内置实现,在我的例子中是依赖于底层数据库的功能的本地类,对于MySQL,使用具有自动增量功能的标识列功能
property: 是描述与数据库表中的列相对应的持久性类属性的一个或多个元素,在我的情况下,它仅由一列组成,因此相应地映射
   该name属性告诉该类中的哪个属性被使用
   该column属性对应于列名在数据库
   该type属性告诉NHibernate数据库列类型,这在大多数情况下应该被自动检测,但是如前所述,可能会有一些麻烦

该映射文件被添加到我的项目根目录中,并且我发现,将XML文件的构建操作设置为“嵌入式资源”,因为这样可以使NHibernate在运行时解析这一点,从而简化了所需的编码当使用映射功能。
所以我们这里将这个配置文件的生成操作设置为嵌入的资源

Paste_Image.png
这个时候我们的配置就完成了 ,下面就可以去实现插入数据

插入数据

            var configuration = new Configuration();
            configuration.Configure();//用来解析nibernate.cfg.xml的,解析数据库的连接文件的
            configuration.AddAssembly("NHibernateController");//添加程序集 解析映射文件 User.hbm.xml

            ISessionFactory sessionFactory = null;
            ISession session = null;
            try
            {
             sessionFactory = configuration.BuildSessionFactory();
               session = sessionFactory.OpenSession();//打开一个跟数据库的回话 ,跟数据库的一个连接
                //添加数据
                User user = new User() { Username = "World" ,Password="1579"};//插入数据World
                session.Save(user);//并且保存起来
            }
            catch (Exception e)
            {
                Console.WriteLine(e);
            }
            finally {
                if (sessionFactory!=null)
                {
                    sessionFactory.Close();
                }
                if (session!=null)
                {
                    session.Close();
                }
            }

这时候我们就可以看到数据已经插入数据库了


Paste_Image.png

事务

那么什么是事务呢,我们来看看,下面画个图理解下


}))KO9957NDPQW%E%JDVC82.png

比如这张图就是一个事务,里面有三个事件,当事件一插入执行成功,事件二由于一些原因没能执行成功,这个时候事件一就要返回成原来的数据,当事件一事件二事件三都执行成功的时候才会去修改数据库中的数据,这就是事务。下面实验下

using NHibernate;
using NHibernate.Cfg;
namespace NHibernateController
{
    class Program
    {

          var configuration = new Configuration();
           configuration.Configure();//用来解析nibernate.cfg.xml的,解析数据库的连接文件的
            configuration.AddAssembly("NHibernateController");//添加程序集 解析映射文件 User.hbm.xml

            ISessionFactory sessionFactory = null;
            ISession session = null;
            ITransaction transaction = null;
            try
            {
             sessionFactory = configuration.BuildSessionFactory();
             session = sessionFactory.OpenSession();//打开一个跟数据库的回话 ,跟数据库的一个连接
          
                //事务
               transaction=  session.BeginTransaction();
                //进行操作
                User user1 = new User() { Username = "EasyA", Password = "1579" };//插入数据
                User user2 = new User() { Username = "EasyB", Password = "544" };

                session.Save(user1);
                session.Save(user2);

                transaction.Commit();//进行提交
            }
            catch (Exception e)
            {
                Console.WriteLine(e);
            }
            finally {
                if (transaction != null)
                {
                    transaction.Dispose();
                }
                if (session != null)
                {
                    session.Close();
                }
                if (sessionFactory!=null)
                {
                    sessionFactory.Close();
                }
            }
      }
}
Paste_Image.png

这个时候是成功插入的,因为条件都满足,但是如果我们插入的第二个数据不成功的话那么第一个数据也不会改变,例如我们数据库中设置了限制用户名不能相等,这个时候我们插入的第二条数据在数据库中已经存在了,那么这个时候第一条数据也会执行失败。数据不会修改。

管理会话工厂(对获取数据库以及增删改查进行一层封装)

那么按照上面那种方法每次都需要去创建Configuration方法会非常麻烦,下面我们去把这个类封装下,首先创建一个NhibernateHelper类,写入以下代码

using NHibernate;
using NHibernate.Cfg;

namespace NHibernateController
{
    class NhibernateHelper
    {
        private static ISessionFactory _sessionFactory;
        private static ISessionFactory SessionFactory
        {
            get
            {
                if (_sessionFactory==null)
                {
                    var configuration = new Configuration();
                    configuration.Configure();
                    configuration.AddAssembly("NHibernateController");

                    _sessionFactory = configuration.BuildSessionFactory();
                }
                return _sessionFactory;
            }

        }

        public static ISession OpenSession()
        {
            return SessionFactory.OpenSession();
        }
    }
}

然后我们还需要创建一个管理User表的文件夹里面创建一个接口IUserManager,一个实现接口的类UserManager,然后我们开始写入

Paste_Image.png
using System.Collections.Generic;
using NHibernateController.Model;

namespace NHibernateController.Manager
{
//接口
using System.Collections.Generic;
using NHibernateController.Model;
namespace NHibernateController.Manager
{
    interface IUserManager
    {
        void Add(User ser);
        void Update(User user);//更新数据
        void Remove(User user); //删除数据
        User GetById(int id); //根据ID获取数据
        User GetByUsername(string username); //根据username获取数据
        ICollection<User> GetAllUsers();  //获取所有数据
        bool VerifyUser(string username, string password);//验证用户密码

    }
}

//实现接口
using System.Collections.Generic;
using NHibernateController.Model;
using NHibernate;
using NHibernate.Criterion;

namespace NHibernateController.Manager
{
    class UserManager : IUserManager
    {
        
        public void Add(User user)
        {
            //也可使用成一个事务
            using (ISession session = NhibernateHelper.OpenSession())
            {
                using (ITransaction transaction = session.BeginTransaction())//事务的开始
                {
                    //进行操作
                    session.Save(user);
                    transaction.Commit();//事物的提交
                }
            }
        }
       
        public ICollection<User> GetAllUsers()
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                //Restrictions.Eq()表示添加查询条件
                // criteria.UniqueResult<User>();得到唯一的结果,返回的是User对象
                IList<User> users = session.CreateCriteria(typeof(User)).List<User>();


                return users;
            }
        }

        public User GetById(int id)//查询条件不会更改数据所以不需要使用事务
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                    //进行操作
                    User user = session.Get<User>(id);//删除数据
                    return user;  
            }
        }
       
        public User GetByUsername(string username)
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                //Restrictions.Eq()表示添加查询条件
                // criteria.UniqueResult<User>();得到唯一的结果,返回的是User对象
                User user = session.CreateCriteria(typeof(User)).Add(Restrictions.Eq("Username", username)).UniqueResult<User>();//创建一个配置文件
                return user;
            }
           
        }
       
        public void Remove(User user)
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                using (ITransaction transaction = session.BeginTransaction())//事务的开始
                {
                    //进行操作
                    session.Delete(user);//删除数据
                    transaction.Commit();//事物的提交
                }
            }
        }
        
        public void Update(User  user)
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                using (ITransaction transaction = session.BeginTransaction())//事务的开始
                {
                    //进行操作
                    session.Update(user);//更新数据
                    transaction.Commit();//事物的提交
                }
            }
        }

        public bool VerifyUser(string username, string password)
        {
            using (ISession session = NhibernateHelper.OpenSession())
            {
                IList<User> users = session.CreateCriteria(typeof(User)).List<User>();
                User user = session.CreateCriteria(typeof(User))
                          .Add(Restrictions.Eq("Username", username))
                          .Add(Restrictions.Eq("Password", password))
                          .UniqueResult<User>();
                if (user == null) return false;
                return true ;
            }
        }
    }
}

这样我们每次直接使用封装好的方法就OK了。非常方便

using System;
using System.Collections.Generic;
using NHibernateController.Model;
using NHibernateController.Manager;

namespace NHibernateController
{
    class Program
    {
        static void Main(string[] args)
        {

            IUserManager userManager = new UserManager();
            ////添加
            //User user = new User() { Id = 8, Username = "sadasd", Password = "15654" };
            //userManager.Add(user);

            ////更新
            //userManager.Update(user);

            ////删除
            //userManager.Remove(user);

            ////根据Id查询值
            //Console.WriteLine(userManager.GetById(8).Password + userManager.GetById(8).Username);

            ////根据Username进行查询
            //Console.WriteLine(userManager.GetByUsername("Sure").Password);

            ////输出所有表里面的数据
            //ICollection<User> users = userManager.GetAllUsers();
            //foreach (User u in users)
            //{
            //    Console.WriteLine(u.Username + " " + u.Password);
            //}

            //验证账号密码是否正确
            Console.WriteLine(userManager.VerifyUser("Sure","157"));

            Console.ReadKey(); 
        }
    }
}

下面是我们查询数据库中所有数据的结果


Paste_Image.png

对照一下用户名和密码都没问题


Paste_Image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,290评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,399评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,021评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,034评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,412评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,651评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,902评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,605评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,339评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,586评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,076评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,400评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,060评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,851评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,685评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,595评论 2 270

推荐阅读更多精彩内容