一、NHibernate 使用配置与实现简单查询

NHibernate是从Hibernate迁移到DotNet来的优秀的ORM框架,它把数据库中的表的关系模型转换为程序中的对象之间的关联模型,从而使程序模型更接近于现实中的逻辑。作为DotNet本身在VS2008 SP1中新增了ADO.NET Entity Framework,但EF非常庞大,远超过ORM的概念范畴,使初学者难以把握。而在VS2005中却没有这种ORM框架,在此我们先以NHibernate为工具研究ORM思想,并研究它如何在VS2005中应用。
ORM就是“对象-映射-关系”的简称,它主要包含三部分:
      关系:关系型数据库。
    对象:实体对象类
    映射:XML文件,指定实体类与表的对应关系。

在这里需要我们操作的就有两点:编写实体对象类,配置映射文件。这两部份一般不需要我们手动编写,我们可以使用CodeSmith的模板来生成。

一、R-关系部份
以下面的数据库为例:
create database mydb
go
use mydb
go
--水果表
create table Fruit
(
Ids varchar(50) primary key,
[Name] varchar(50) not null,
Price decimal(8,2),
Source varchar(50),
Stack varchar(50),
Numbers int, --库存数量
[Image] varchar(50)
)
go
insert into fruit values('k001','苹果',2.4,'烟台','2',100,'image/0.gif')
insert into fruit values('k002','菠萝',1.4,'广东','3',100,'image/1.gif')
insert into fruit values('k003','桔子',2.4,'福州','3',100,'image/2.gif')
insert into fruit values('k004','葡萄',2.4,'新缰','2',100,'image/3.gif')
insert into fruit values('k005','樱桃',2.4,'青岛','4',100,'image/4.gif')
insert into fruit values('k006','桃子',2.4,'花果山','5',100,'image/5.gif')
insert into fruit values('k007','香蕉',2.4,'济南','5',100,'image/6.gif')

Fruit表是一个独立的单一表,我们以此表为例,研究如何配置ORM框架,并实现增、删、改、查功能

二、O-对象部份
第一步:新建一个类库Common
第二步:向Common项目中添加NHibernate.dll引用
第三步:使用CodeSmith执行NHibernate.cst文件生成实体类和映射文件
《图1》
Fruit.cs的文件代码如下:
using System;
using System.Collections;
using System.Web.UI.WebControls;

使用配置与实现简单查询" alt="一、NHibernate  使用配置与实现简单查询" src="http://hiphotos.baidu.com/grayworm/pic/item/3be3a1f8273c4211d8f9fd52.jpg" border=0 SMALL="0">

 

namespace Common
{

public class Fruit : System.IComparable
{
     //成员变量
   protected string _id;
   protected string _name;
   protected decimal _price;
   protected string _source;
   protected string _stack;
   protected int _numbers;
   protected string _image;
   protected static String _sortexpression_r = "Id";
   protected static SortDirection _sortDirection = SortDirection.Ascending;

        //构造函数
   public Fruit() { }

   public Fruit( string name, decimal price, string source, string stack, int numbers, string image )
   {
    this._name = name;
    this._price = price;
    this._source = source;
    this._stack = stack;
    this._numbers = numbers;
    this._image = image;
   }
  
   //属性
   public virtual string Id
   {
    get {return _id;}
    set
    {
     if ( value != null && value.Length > 50)
      throw new ArgumentOutOfRangeException("Invalid value for Id", value, value.ToString());
     _id = value;
    }
   }

   public virtual string Name
   {
    get { return _name; }
    set
    {
     if ( value != null && value.Length > 50)
      throw new ArgumentOutOfRangeException("Invalid value for Name", value, value.ToString());
     _name = value;
    }
   }

   public virtual decimal Price
   {
    get { return _price; }
    set { _price = value; }
   }

   public virtual string Source
   {
    get { return _source; }
    set
    {
     if ( value != null && value.Length > 50)
      throw new ArgumentOutOfRangeException("Invalid value for Source", value, value.ToString());
     _source = value;
    }
   }

   public virtual string Stack
   {
    get { return _stack; }
    set
    {
     if ( value != null && value.Length > 50)
      throw new ArgumentOutOfRangeException("Invalid value for Stack", value, value.ToString());
     _stack = value;
    }
   }

   public virtual int Numbers
   {
    get { return _numbers; }
    set { _numbers = value; }
   }

   public virtual string Image
   {
    get { return _image; }
    set
    {
     if ( value != null && value.Length > 50)
      throw new ArgumentOutOfRangeException("Invalid value for Image", value, value.ToString());
     _image = value;
    }
   }

        public static String Sortexpression_r
        {
            get { return _sortexpression_r; }
            set { _sortexpression_r = value; }
        }

        public static SortDirection SortDirection
        {
            get { return _sortDirection; }
            set { _sortDirection = value; }
        }
  
        //实现IComparable接口的排序方法
        public virtual int CompareTo(object obj)
        {
    if (!(obj is Fruit))
     throw new InvalidCastException("This object is not of type Fruit");
   
    int relativue;
    switch (Sortexpression_r)
    {
     case "Id":
      relativue = this.Id.CompareTo(((Fruit)obj).Id);
      break;
     case "Name":
      relativue = this.Name.CompareTo(((Fruit)obj).Name);
      break;
     case "Price":
      relativue = (this.Price != null) ? this.Price.CompareTo(((Fruit)obj).Price) : -1;
      break;
     case "Source":
      relativue = (this.Source != null) ? this.Source.CompareTo(((Fruit)obj).Source) : -1;
      break;
     case "Stack":
      relativue = (this.Stack != null) ? this.Stack.CompareTo(((Fruit)obj).Stack) : -1;
      break;
     case "Numbers":
      relativue = (this.Numbers != null) ? this.Numbers.CompareTo(((Fruit)obj).Numbers) : -1;
      break;
     case "Image":
      relativue = (this.Image != null) ? this.Image.CompareTo(((Fruit)obj).Image) : -1;
      break;
                default:
                    goto case "Id";
    }
            if (Fruit.SortDirection == SortDirection.Ascending)
     relativue *= -1;
    return relativue;
   }
}
}

三、M-映射部分
Fruit.hbm.xml映射文件如下(在这里我用的是nhibernate2.2):
<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
    <!--Common.Fruit类与Fruit表进行映射-->
<class name="Common.Fruit, Common" table="Fruit">
     <!--主键Id映射到Ids列-->
   <id name="Id" type="String" unsaved-value="null">
    <column name="Ids" length="50" sql-type="varchar" not-null="true" unique="true" index="PK__Fruit__7C8480AE"/>
   <generator class="native" />
   </id>
   <!--列映射-->
   <property name="Name" type="String">
    <column name="Name" length="50" sql-type="varchar" not-null="true"/>
   </property>
   <property name="Price" type="Decimal">
    <column name="Price" length="5" sql-type="decimal" not-null="false"/>
   </property>
   <property name="Source" type="String">
    <column name="Source" length="50" sql-type="varchar" not-null="false"/>
   </property>
   <property name="Stack" type="String">
    <column name="Stack" length="50" sql-type="varchar" not-null="false"/>
   </property>
   <property name="Numbers" type="Int32">
    <column name="Numbers" length="4" sql-type="int" not-null="false"/>
   </property>
   <property name="Image" type="String">
    <column name="Image" length="50" sql-type="varchar" not-null="false"/>
   </property>
</class>
</hibernate-mapping>
第四步:把Fruit.hbm.xml文件作为嵌入资源,否则找不到该文件,就无法识别我们的实体对象类。

到此为至,我们的O、R、M三部份都实现完成了,下面的内容就是配置NHibernate引擎。
第五步:编写hibernate.cfg.xml配置文件,并把该文件入在网站的BIN文件夹中,否则也会找不到该文件。

四、NHibernate的配置部份
<?xml version="1.0" encoding="utf-8"?>
<hibernate-configuration xmlns="urn:nhibernate-configuration-2.2" >
    <session-factory>
        <property name="connection.driver_class">NHibernate.Driver.SqlClientDriver</property>
        <property name="connection.connection_string">
            server=.;database=mydb;uid=sa;pwd=123;
        </property>
        <property name="adonet.batch_size">10</property>
        <property name="show_sql">true</property>
        <property name="dialect">NHibernate.Dialect.MsSql2005Dialect</property>
        <property name="use_outer_join">true</property>
        <property name="command_timeout">10</property>
        <property name="query.substitutions">true 1, false 0, yes 'Y', no 'N'</property>
        <mapping assembly="Common"/>
    </session-factory>
</hibernate-configuration>
hibernate.cfg.xml中配置了NHibernate的数据库链接,Sql方言相关信息。

到目前为止所有的配置信息都已完成,剩下的内容就是编写代码访问数据库了。
NHibernate访问数据库常用的有两个接口:ISessionFactory和ISession。
ISessionFactory接口:负责初始化NHibernate。它充当数据存储源的代理,并负责创建Session对象。这里用到了工厂模式。一般情况下,一个项目通常只需要一个SessionFactory就够,当需要操作多个数据库时,可以为每个数据库指定一个SessionFactory。
Session接口:负责执行被持久化对象的CRUD操作(增、删、改、查)。但ISession是非线程安全的。NHibernate的ISession不同于HTTP应用中的HttpSession,我们将HttpSesion对象称为用户session。

五、使用NHibernate访问数据库
第六步:在Common项目中建立SessionManager类
using System;
using System.Collections.Generic;
using System.Text;
using NHibernate;
using NHibernate.Cfg;
namespace Common
{
    public class SessionManager
    {
        private ISessionFactory _SessionFactory;
        private ISession _Session;
        public SessionManager()
        {
            //生成SessionFactory
            _SessionFactory = new Configuration().Configure().BuildSessionFactory();
            //使用SessionFactory对象生成Session对象
            _Session = _SessionFactory.OpenSession();
        }
        //返回Session对象
        public ISession Session
        {
            get
            {
                return _Session;
            }
        }
    }
}
该类的作用就是生成Session对象,以便对数据库的操作。

第七步:创建数据库访问代码
using System;
using System.Collections.Generic;
using System.Text;
using NHibernate;

namespace Common
{
    public class FruitDA
    {
        private ISession _Session;
        public FruitDA()
        {
            //生成ISession对象
            _Session = new SessionManager().Session;
        }
        public IList<Fruit> Select()
        {
            //使用ISession对象的CreateQuery()方法来执行HQL查询数据库,并返回泛型集合数据
            IList<Fruit> list = _Session.CreateQuery("from Fruit").List<Fruit>();
            return list;
        }
    }
}

第八步:页面调用FruitDA类的Select()方法查询数据,并显示在页面上
GridView1.DataSource = new FruitDA().Select();
GridView1.DataBind();

好了,使用NHibernate建立一个简单查询就完成了。

NHibernate与ADO.NET访问数据库
NHibernate只需要我们编写好实体类和映射文件就可以了,当然这部份代码我们可以使用CodeSmith来生成,一旦配置好NHibernate后,所以的数据库访问都由NHibernate代我们完成。作为开发人员,我们只需要理清商业逻辑,管理好我们的实体对象就可以了,不用再使用SqlConnection,SqlCommand等对象去为每个表编写枯燥数据库操作类了。

NHibernate与Linq To Sql
从易用性上来讲,NHibernate确实不如Linq To Sql容易使用,在实体类的生成和映射文件的生成中Linq To Sql只需要点点鼠标就可以实现了。但Linq To Sql是在VS2008中提供出来的,在VS2005中无法使用的。另外,对数据量不太大的情况下,Linq To Sql的性能不如NHibernate高,在数据量大的情况下,Linq To Sql则表现出强悍的性能。另外Linq To Sql目前只适用于SqlServer数据库,还不能算是一个完善的ORM框架。

你可能感兴趣的:(Hibernate)