随笔-81  评论-910  文章-0  trackbacks-56

注:本文中使用的是NH1.0.2版本,其他版本配置可能略有不同。

示例代码下载


数据库脚本下载


一、体系结构

首先我们来看看NHibernate的体系结构:

NHibernate是数据库和应用程序之间的持久化层,应用程序和NHibernate通过持久化(Persistent)对象联系在一起。NHibernate使用app.config或者web.config和映射文件进行设置。

下面来看看怎么使用NHibernate API的一个最小子集。

下面是图中一些对象的定义:

Configuration(NHibernate.Cfg.Configuration)

配置对象,用来指定用来出创建ISessionFactory的属性和映射文件。(图中没有此对象)
SessionFactory (NHibernate.ISessionFactory)

Session的工厂。

会话,Session (NHibernate.ISession)

代表应用程序和持久化层之间的一次对话。封装了一个ADO.NET连接。也是Transaction的工厂。保存有必需的(第一级)持久化对象的缓存,用于遍历对象图,或者通过标识符查找对象。

事务Transaction (NHibernate.ITransaction)

(可选) 单线程,生命期短促的对象,应用程序用它来表示一批工作的原子操作。是底层的ADO.NET事务的抽象。一个Session某些情况下可能跨越多个Transaction 事务。

 

二、示例代码

我们看看他们是如何协同工作的,请看下面的代码。

代码示例(未改变数据库中的数据,查询操作):

//配置Configuration
Configuration cfg = new Configuration().Configure();
//创建ISessionFactory
ISessionFactory factory = cfg.BuildSessionFactory();
//打开ISession
ISession session = factory.OpenSession();
try
{
    在这里添加操作
}

finally
{
    session.Close();
}

Configuration的Configure方法,将使用应用程序的配置文件或者hibernate.cfg.xml文件的<hibernate-configuration>节点进行配置,返回值是一个配置好了的Configuration对象的实例。

然后这个Configuration通过BuildSessionFactory方法创建ISessionFactory的实例。

ISessionFactory的实例通过OpenSession打开一个ISession的实例,用这个实例完成数据库的操作,然后关闭它。

如果你的代码中需要事务,只需要把代码稍微修改一下。

代码示例(改变了数据库中的数据,数据的增删改):

//配置Configuration
Configuration cfg = new Configuration().Configure();
//创建ISessionFactory
ISessionFactory factory = cfg.BuildSessionFactory();
//定义事务
ITransaction tx = null;
//打开ISession
ISession session = factory.OpenSession();
try
{
    
//开始事务
    tx = session.BeginTransaction();
    在这里添加操作
    tx.Commit();
}

catch(HibernateException ex)
{
    
if (tx!=null) tx.Rollback();
    
throw ex;
}

finally
{
    
//关闭ISession
    session.Close();
}

我一般在不影响数据的方法(例如:查询)中不包含事务,而影响数据的方法(例如:增删改)使用事务。

 

三、配置文件

前面提到Configuration的Configure方法需要用到应用程序的配置文件或者hibernate.cfg.xml文件,这里我使用hibernate.cfg.xml配置文件,在下一章我们再探讨使用应用程序的配置文件和其他的配置方法。

这是一个典型的配置文件:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-configuration xmlns="urn:nhibernate-configuration-2.0">
    
<session-factory name="DDLLY.MyDoc.NHibernateTest.QuickStart">
        
<!-- 属性 -->
        
<property name="connection.provider">NHibernate.Connection.DriverConnectionProvider</property>
        
<property name="connection.driver_class">NHibernate.Driver.SqlClientDriver</property>
        
<property name="connection.connection_string">server=.;database=user;uid=sa;pwd=App1234;</property>
        
<property name="show_sql">false</property>
        
<property name="dialect">NHibernate.Dialect.MsSql2000Dialect</property>
        
<property name="use_outer_join">true</property>
        
<property name="query.substitutions">true 1, false 0, yes 'Y', no 'N'</property>
        
<!-- 映射文件 -->
        
<mapping assembly="DDLLY.MyDoc.NHibernateTest.QuickStart" />
    
</session-factory>
</hibernate-configuration>

属性connection.provider设置了数据库连接提供者(一般不需要更改,除非你决定自己实现提供者)。

connection.driver_class设置了数据库的的驱动类(这里设置了SqlServer的驱动类)。

connection.connection_string是数据库的连接字符串。

show_sql设置是否在运行时是否在控制台显示正在执行的SQL语句(这个属性在调试程序时很有用)。

dialect是NHibernate方言, 可以让NHibernate使用某些特定的数据库平台的特性。

use_outer_join在后面的章节会有介绍。

query.substitutions,把NHibernate查询中的一些短语替换为SQL短语(比如说短语可能是函数或者字符)。

mapping节点设置了将使用到的映射文件,现在的设置方法是使用"DDLLY.MyDoc.NHibernate.QuickStart“中所有的映射文件。

 

四、映射文件

ISession怎么知道和数据库的哪张表哪个字段打交道,以及怎么把表和实体对象联系起来呢?靠的就是映射文件。

假设我们有个很简单的例子,我们有如下的一张表需要进行数据操作(字段和表含义一看就知道,我就不罗嗦了):

CREATE TABLE [dbo].[users] (
[Id] [int] IDENTITY (1, 1) NOT NULL ,
[UserName] [nvarchar] (64)  NOT NULL ,
[Password] [varchar] (32)  NOT NULL ,
[Email] [varchar] (64)  NULL
)

我们设置它的映射文件和相应的代码文件:

<?xml version="1.0" encoding="utf-8" ?> 
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.0">
    
<class name="DDLLY.MyDoc.NHibernateTest.QuickStart.User, DDLLY.MyDoc.NHibernateTest.QuickStart" table="users">
        
<id name="Id" type="Int32">
            
<generator class="identity" />
        
</id>
        
        
<property name="UserName" type="String"/>
        
<property name="Password" type="String"/>
        
<property name="Email" type="String"/>
    
</class>    
</hibernate-mapping>




 
namespace DDLLY.MyDoc.NHibernateTest.QuickStart
{
    
/// <summary>
    
/// User实体
    
/// </summary>

    public class User
    
{
        
public User()
        
{
        }


        
private int m_Id;
        
private string m_UserName;
        
private string m_Password;
        
private string m_Email;

        
public int Id
        
{
            
get return m_Id; }
            
set { m_Id = value; }
        }


        
public string UserName
        
{
            
get return m_UserName; }
            
set { m_UserName = value; }
        }


        
public string Password
        
{
            
get return m_Password; }
            
set { m_Password = value; }
        }


        
public string Email
        
{
            
get return m_Email; }
            
set { m_Email = value; }
        }

    }

}

<class name="DDLLY.MyDoc.NHibernate.QuickStart.User, DDLLY.MyDoc.NHibernate.QuickStart" table="users"> 表示映射文件对应的类文件是DDLLY.MyDoc.NHibernate.QuickStart程序集中的 DDLLY.MyDoc.NHibernate.QuickStart.User类,逗号后面是表示程序集。对应的数据库表为users。

<id name="Id" type="Int32">
<generator class="identity" />
</id>

表示Id是主键字段,generator是生成器,这里是使用SqlServer中内置标识字段来生成。

property定义了其他的字段。

 

五、完成代码

好了,下面我们可以开始对数据进行操作了。

我们来看看ISession为我们提供的常用方法。

Load:读取实体对象

Find:通过HQL查找实体对象

Save:保存实体对象

Delete:删除实体对象

用这些方法我们便可以完成对数据库的增删改查的操作了。下面是我完成的代码(使用NUnit进行测试)。

using System.Collections;
using System.Data.SqlClient;
using NHibernate;
using NHibernate.Cfg;
using NUnit.Framework;

namespace DDLLY.MyDoc.NHibernateTest.QuickStart
{
    
/// <summary>
    
/// User测试类
    
/// </summary>

    [TestFixture]
    
public class UserFixture
    
{
        
常量

        
SetUp和TearDown

        
私有方法

        
测试方法
    }

}
posted on 2005-12-12 18:06 DDL 阅读(9948) 评论(38)  编辑 收藏 网摘 所属分类: NHibernate

评论:
#1楼 2005-12-12 23:49 | bing7788[未注册用户]
谢谢了,为我们初学者办了一件好事、实事!
  回复  引用    
#2楼 2005-12-15 03:32 | 周奔驰      
简洁易懂
  回复  引用  查看    
#3楼 2005-12-16 21:34 | mzl[未注册用户]
好极啦!
  回复  引用    
#4楼 2005-12-23 13:42 | vb1980[未注册用户]
好文,希望尽快写后面的内容。
会常来看看的。

  回复  引用    
#5楼 2005-12-28 16:20 | 路过[未注册用户]
用这个东西有什么好处吗?
看代码还是用了所谓的“HQL”,跟直接用SQL操作有什么好处呢?

  回复  引用    
#6楼 2006-01-23 15:23 | Neo(淡泊江湖)[未注册用户]
和SPL(http://tintown.cnblogs.com/category/12787.html)有什么区别?
  回复  引用    
#7楼 2006-02-20 14:25 | 路过[未注册用户]
我用vs.net2005,楼主的示例运行不了,单元测试不能通过,错误代码如下,请指教:
测试方法 TestProject1.UserFixtureTest.TestLoadUserTest 引发异常: NHibernate.HibernateException: problem parsing configuration : System.IO.FileNotFoundException: 未能找到文件“C:\Documents and Settings\qy\My Documents\Visual Studio 2005\Projects\QuickStart\TestResults\qy_QIAN 2006-02-20 13_56_55\Out\hibernate.cfg.xml”。

  回复  引用    
#8楼[楼主] 2006-02-22 09:36 | DDL      
@路过
把Configuration cfg = new Configuration().Configure()改为
Configuration cfg = new Configuration().Configure("hibernate.cfg.xml的绝对路径")可执行

  回复  引用  查看    
#9楼 2006-02-22 10:34 | 路过[未注册用户]
谢谢,单元测试通过了
  回复  引用    
#10楼 2006-04-02 13:33 | Michael      
我在使用ISession的Find方法时,出现下面的错误:
in expected: <end-of-text> (possibly an invalid or unmapped class name was used in the query) [from N_USER where USERNAME='LI']

请问这是怎么回事,谢谢!

  回复  引用  查看    
#11楼 2006-04-24 23:22 | 李天[未注册用户]
谢谢,不错。。
  回复  引用    
#12楼 2006-08-18 17:07 | zjf[未注册用户]
用NHibernate有何好处!
  回复  引用    
#13楼 2006-08-24 09:58 | Tseng      
@DDL
用例编译通过,但是用NUit(2.0)测试不通过。
Error Message:
DDLLY.MyDoc.NHibernateTest.QuickStart.UserFixture.TestAddUser : NHibernate.ADOException : cannot open connection
----> NHibernate.ADOException : Could not create connection from Driver
----> System.Data.SqlClient.SqlException : Login failed for user 'sa'.

我的数据库,用户名和密码都是正确的。不知道什么原因?

  回复  引用  查看    
#14楼[楼主] 2006-08-24 10:29 | DDL      
@Tseng
应该是你的密码不对,修改你配置文件中的连接字符串

  回复  引用  查看    
#15楼 2006-08-24 11:31 | Tseng      
@DDL
没理由呀,因为我是用我重新改过来的密码(123456就那么简单)的也是报同样的错误。

  回复  引用  查看    
#16楼 2006-08-24 11:37 | Tseng      
@DDL
我修改的是nhibernate.cfg.xml,更奇怪的是,我把uid改成fsdfsdf(乱写的),也是报同样的错误;或者我修改database的sa密码为空,也是一样错误。

  回复  引用  查看    
#17楼[楼主] 2006-08-24 11:42 | DDL      
@Tseng
请改\bin\Debug或者\bin\Release里面的配置文件.
new Configuration().Configure()调用的是和应用程序同目录下的配置文件

  回复  引用  查看    
#18楼 2006-08-24 11:58 | Tseng      
@DDL
OK,谢谢DLL。还有个问题,就是编译的时候为什么没有把nhibernate.cfg.xml copy过去?

  回复  引用  查看    
#19楼[楼主] 2006-08-24 12:17 | DDL      
@Tseng
据我所知编译的时候是不会copy过去的,只能自己copy

  回复  引用  查看    
#20楼 2006-09-04 11:30 | gothing[未注册用户]
可以拷贝的,在项目属性的Build Events下,在Post-build event command line中输入命令行(vs2005):copy $(ProjectDir)\hibernate.cfg.xml $(TargetDir)
  回复  引用    
#21楼 2006-09-14 10:59 | Pobear[未注册用户]
警告 1 未能将文件“bin\Debug\hibernate.cfg.xml”添加到项目中。项目中已存在具有相同路径的文件。 QuickStart

Why?
Thanks

  回复  引用    
#22楼 2006-09-20 16:48 | liaokobe[未注册用户]
@Michael
我也碰到了同样的问题,还没有找到解决方法。

  回复  引用    
#23楼 2007-01-18 23:44 | Seven[未注册用户]
@Tseng
在 hibernate.cfg.xml文件的属性里的 "复制到输出目录"选项选成"如果较新则复制"或者"始终复制"

  回复  引用    
#24楼 2007-03-06 14:04 | teo[未注册用户]
NHibernate.MappingException: element was not found in the configuration file. at NHibernate.Cfg.Configuration.DoConfigure(XmlDocument doc) at NHibernate.Cfg.Configuration.Configure(XmlTextReader reader) at NHibernate.Cfg.Configuration.Configure(String fileName) at WebApplication2.WebForm1.Page_Load(Object sender, EventArgs e) in c:\inetpub\wwwroot\webapplication2\webform1.aspx.cs:line 109
  回复  引用    
#25楼 2007-03-06 14:11 | teo[未注册用户]
NHibernate.MappingException: element was not found in the configuration file. at NHibernate.Cfg.Configuration.DoConfigure(XmlDocument doc) at NHibernate.Cfg.Configuration.Configure(XmlTextReader reader) at NHibernate.Cfg.Configuration.Configure(String fileName) at WebApplication2.WebForm1.Page_Load(Object sender, EventArgs e) in c:\inetpub\wwwroot\webapplication2\webform1.aspx.cs:line 109. 搞了半天还是没有头绪, 请高手帮帮忙。。。 多谢了!
  回复  引用    
#26楼 2007-03-06 18:09 | coolstr[占卜師oοΟ]      
ddlly的 NHibernate快速起步 ,里面的代码,出现如下情况!

NHibernate.MappingException: Unknown entity class: NHibernate.Examples.QuickStart.User

是什么原因啊?

  回复  引用  查看    
#27楼 2007-03-26 15:45 | fatcherub[未注册用户]
Try to configure the .hbm.xml file's Build Action to 'Embedded Resource'
  回复  引用    
#28楼 2007-04-18 11:11 | Aloha_Ajax[未注册用户]
how to configure the file's Build Action proporty ? thx
  回复  引用    
#29楼 2007-04-18 11:13 | Aloha_Ajax[未注册用户]
additional , in Visual Studio 2005 environment !~
  回复  引用    
#30楼 2007-05-11 22:08 | chenwj[未注册用户]
谢谢,让我完成第一个 NHibernate 程序,实现增删改查。
  回复  引用    
#31楼 2007-06-03 19:15 | 菜鸟[未注册用户]
D:\Inetpub\wwwroot\NHibernate\ClassLibrary1\Class1.cs(4): 类型或命名空间名称“Cfg”在类或命名空间“NHibernate”中不存在(是否缺少程序集引用?)
可我以添加了NHibernate.dll的引用啦,还要添加什么东东呀。请问这个问题要怎么解决3Q

  回复  引用    
#32楼 2008-03-25 16:38 | jazz[未注册用户]
楼主,我是初学者,请问下载了你的示例后怎么用?怎么进行单元测试?问题很垃圾,不过还是希望楼主或者高手帮忙。在此先谢过了。
  回复  引用    
#33楼 2008-11-18 08:49 | 幻蓝      
和JAVA的Hibernate类似....

谢啦

  回复  引用  查看    
#34楼 2009-03-03 13:19 | fengzhen[未注册用户]
示例代码下来之后,在本机运行,总是报这样,那样的错误
1,未能加载文件或程序集“log4net, Version=1.2.9.0, Culture=neutral, PublicKeyToken=b32731d11ce58905”
这个跟Log4net有什么联系?
2,配置文件中的配置老是出错:找不到节点<session-factory xmlns='urn:nhibernate-configuration-2.2'> element was not found in the configuration file."
这个节点明明存在的
后来将命名空间xmlns='urn:nhibernate-configuration-2.2'>
加到<session-factory name="DDLLY.MyDoc.NHibernate.QuickStart" xmlns='urn:nhibernate-configuration-2.2'>
还是出错
真是快崩溃了!!

  回复  引用    
#35楼 2009-03-13 16:02 | fengzhen[未注册用户]
运行到:
User user;
//创建ISessionFactory
ISessionFactory factory = cfg.BuildSessionFactory();

时出现在以下BUG

The following types may not be used as proxies:
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method get_Email should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method set_UserName should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method get_Password should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method set_Id should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method set_Password should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method get_Id should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method get_UserName should be virtual
DDLLY.MyDoc.NHibernateTest.QuickStart.User: method set_Email should be virtual

请问出现这样的Bug是什么原因?
Thanks!

  回复  引用    
#36楼 2009-05-12 16:51 | 魔方网[未注册用户]
thanks!
http://www.mofun.cc

  回复  引用    
#37楼 2009-05-30 11:16 | eastcowboy      
单元测试时出现如下错误
TestNHibernate.UserFixture.TestLoadUser:
NHibernate.MappingException : No persister for: TestNHibernate.Users
是不是要具体指明映射文件的位置
.cfg.xml中mapping节点就指明程序集,它能自动加载里面的mapping文件吗

  回复  引用  查看    



发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 295712 LxB1yNJjCto=



相关文章:

相关链接: