本节内容
- 引入
- SchemaExport工具
- SchemaUpdate工具
- 实例分析
- 结语
引入
我其实都是一直先编写持久化类和映射文件,然后使用SchemaExport工具生成数据库架构。这样的方式就是领域驱动设计/开发(DDD,Domain Driven Design/Development)。我的理解是系统的设计应该基于对象模型,主要考虑对象的设计和逻辑上,然后按照对象模型建立数据库关系模型,这才是现在面向对象开发的步骤,并不是上一篇先设计数据库然后再设计对象。用一幅图可以形象的说明领域驱动设计:
当在设计时,我们的领域模型需要改变,只需修改NHibernate结构和应用程序,不需要修改数据库架构,只要利用SchemaExport工具重新生成数据库架构就可以了。但是使用数据库只是其中一种方式,我们也可以使用XML文件来存储数据。
SchemaExport工具
NHibernate的hbm2dll提供SchemaExport工具:给定一个连接字符串和映射文件,不需输入其他东西就可以按照持久化类和映射文件自动生成数据库架构,现在SchemaExport工具还不是很强大,但是一般应用足够了,它还是一个相当原始的API还在不断改进。
SchemaExport工具就是把DDL脚本输出到标准输出,同时/或者执行DDL语句。SchemaExport工具提供了三个方法,分别是Drop()、Create()、Execute(),前两个方法实质是调用Execute()方法。通常使用Execute()方法来生成数据库架构的。
SchemaUpdate工具
在NHibernate2.0中新添加SchemaUpdate工具,可以用来更新数据库架构。但是我觉得没有什么作用,因为它不能Drop现有的表或列,也不能更新现有的列,只能添加新的表和列。如果我需要删除表或者列或者修改其中列,SchemaUpdate工具就显得无能为力了。
实例分析
知道了上面的知识就好好实战一下:看看具体怎么使用呢?
1.SchemaExport工具实战
通常我们使用生成数据库架构代码实例像这样:
Configuration cfg=new Configuration(); cfg.Configure(); SchemaExport export =new SchemaExport(cfg); export.Execute(....);
1.准备工作
现在数据访问测试层新建一SchemaExportFixture.cs文件用于测试生成实战。声明一个全局变量_cfg,编写[SetUp]方法在每个测试方法执行之前调用:
[TestFixture] public class SchemaExportFixture { private Configuration _cfg; [SetUp] public void SetupContext() { _cfg = new Configuration(); _cfg.Configure(); } //测试...... }
2.测试Drop(script, export)方法
[Test] public void DropTest() { var export = new SchemaExport(_cfg); export.Drop(true, true); }
Drop(script, export)方法根据持久类和映射文件执行删除数据库架构。有两个参数,第一个为True就是把DDL语句输出到控制台,第二个为True就是根据持久类和映射文件执行删除数据库架构操作,经过调试可以发现Drop(script, export)方法其实质是执行了Execute(script, export, true, true)方法。
3.测试Create(script, export)方法
[Test] public void CreateTest() { var export = new SchemaExport(_cfg); export.Create(true, true); }
Create(script,export)方法根据持久类和映射文件先删除架构后创建删除数据库架构。有两个参数,第一个为True就是把DDL语句输出到控制台,第二个为True就是根据持久类和映射文件先执行删除再执行创建操作,经过调试可以发现这个方法其实质是执行Execute(script,export, false, true)方法。
4.测试Execute(script, export, justDrop, format)方法
[Test] public void ExecuteTest() { var export = new SchemaExport(_cfg); export.Execute(true, true, false, false); }
Execute(script, export, justDrop, format)方法根据持久类和映射文件先删除架构后创建删除数据库架构。有四个参数,第一个为True就是把DDL语句输出到控制台;第二个为True就是根据持久类和映射文件在数据库中先执行删除再执行创建操作;第三个为false表示不是仅仅执行Drop语句还执行创建操作,这个参数的不同就扩展了上面两个方法;第四个参数为false表示不是格式化输出DDL语句到控制台,是在一行输出的。
所谓格式化输出就像这样:
一行输出就像这样:
5.测试Execute(script, export, justDrop, format, connection, exportOutput)方法
[Test] public void ExecuteOutTest() { var export = new SchemaExport(_cfg); var sb = new StringBuilder(); TextWriter output = new StringWriter(sb); export.Execute(true, false, false, false, null, output); }
Execute(script, export, justDrop, format, connection, exportOutput)方法根据持久类和映射文件先删除架构后创建删除数据库架构。有六个参数,第一个为True就是把DDL语句输出到控制台;第二个为false就是不执行DDL语句;第五个为自定义连接。当export为true执行语句时必须打开连接。该方法不关闭连接,null就是使用默认连接,最后一个参数自定义输出,这里我输出到TextWriter中。
2.SchemaUpdate工具实战
现在数据访问测试层新建一SchemaUpdateFixture.cs文件用于测试生成实战。先声明一个全局变量_cfg:
private Configuration _cfg;
这里我用另外一种方式配置映射文件,先定义两个映射XML分别代表旧的和新的这样才能体现测试更新数据库架构的意义。
旧映射XML:这里我使用Product持久化类,由于在之前我们定义了Product持久化类,这里直接模拟定义映射XML:拥有主键ProductId和Name字段。
public const string product_xml = "<?xml version='1.0' encoding='utf-8' ?>" + "<hibernate-mapping xmlns='urn:nhibernate-mapping-2.2'" + " assembly='DomainModel'" + " namespace='DomainModel'>" + " <class name='DomainModel.Entities.Product,DomainModel'>" + " <id name='ProductId'>" + " <generator class='native'/>" + " </id>" + " <property name='Name'/>" + " </class>" + "</hibernate-mapping>";
新映射XML:更新上面映射XML:主键ProductId(没有改变);Name字段:添加不可为空和长度为50;另外增加了Cost字段,类型为float不可为空。
public const string newproduct_xml = "<?xml version='1.0' encoding='utf-8' ?>" + "<hibernate-mapping xmlns='urn:nhibernate-mapping-2.2'" + " assembly='DomainModel'" + " namespace='DomainModel'>" + " <class name='DomainModel.Entities.Product,DomainModel'>" + " <id name='ProductId'>" + " <generator class='native'/>" + " </id>" + " <property name='Name' not-null='true' length='50' />" + " <property name='Cost' type='float' not-null='true'/>" + " </class>" + "</hibernate-mapping>";
测试前利用旧映射XML创建数据库架构:使用[SetUp]在测试前执行,按照旧映射XML创建数据库架构并格式化输出DDL语句:
[SetUp] public void SetupContext() { //模拟旧系统 _cfg = new Configuration(); _cfg.Configure(); _cfg.AddXml(product_xml); var export = new SchemaExport(_cfg); export.Execute(true, true, false, true); }
测试更新数据库架构:使用SchemaUpdate类提供的唯一的Execute(script, doUpdate)方法按照新映射XML更新数据库架构:
[Test] public void UpdateExistingDatabaseSchemaTest() { _cfg = new Configuration(); _cfg.Configure().AddXml(newproduct_xml); var update = new SchemaUpdate(_cfg); update.Execute(true, true); }
测试输出结果如图所示,如果你觉得不放心再看看数据库Product表。
看到了吗?这显然不是我要求的,首先按照旧映射XML创建了数据库架构,但是更新数据库架构显得无能为力,仅仅增加了Cost字段,我想更新Name字段属性为不可为空和长度为50,但是SchemaUpdate工具不能做到!我觉得这个类目前还没有什么作用,期待下一个版本来完善。
结语
这篇文章通过实例介绍NHibernate中提供两个实用工具SchemaExport工具利用持久化类和映射文件生成数据库架构。SchemaUpdate工具通过持久化类和映射文件更新数据库架构。
相关推荐
NHibernate是一个面向.NET环境的对象/关系数据库映射工具。对象/关系数据库映射(object/relational mapping (ORM))这个术语表示一种技术,用来把对象模型表示的对象映射到基于SQL的关系模型数据结构中去。 ...
SQLite 与 NHibernate 在 .NET 程序中使用.doc
本文档是我在使用NHibernate框架的一些感悟,里面是我精心炮制的带图详细教程,一步一步教导初学者使用NHibernate框架,适合初学者,大家可以交流一下心得。
在.Net中使用SQLite,还需要一个针对SQLite的ADO.NET Provider,即需要sqlite.net.dll 一切准备就绪后,开始,编译运行,出现问题了,报一个什么finsalConnection无法转变为IDBConnection, 在网上查资料,搞了半天,网上说...
MVC+NHibernate+Spring.net整合 http://znieyu.cnblogs.com/
使用nhibernate,spring.net搭建的平台,sqlserver2008,当做学习之用,有什么改进之处,请提出
刘冬编写的Spring.NET,NHibernate,ASP.NET MVC例子 原文:http://www.cnblogs.com/GoodHelper/archive/2009/11/19/SpringNet_Nhibernate_AspNetMvc.html
C#.net 使用NHibernate做持久层,spring.net 进行DI,Codesmith作为代码生成工具
NHibernate ASP.NET 企业级应用案例。全英文。案例所用的框架很明了也很经典,还应用了很多设计模式。
再次向对NHibernate陌生的朋友介绍一下NHibernate,它是一个面向.NET 环境的对象/关系数据库映射工具。用来把对象映射到基于SQL 的关系模型数据结构中去。因此本书围绕这些主要讲解NHibernate是什么、如何建立第一个...
NHibernate 最近发布了 5.1.3 版本, 支持 .NET Standard 2.0 , 这意味着可以在 .NET Core 2.0 应用中使用, 本文就已 WebAPI 应用为例, 介绍一下如何在 .NET Core 应用中如何使用 NHibernate 。下面话不多说了,...
刘冬编写的Spring.NET整合NHibernate例子 原文:http://www.cnblogs.com/GoodHelper/archive/2009/11/18/SpringNet_NHibernate.html
NHibernate ASP.NET 企业级应用案例。英文。网上找到的。案例所用的框架很明了也很经典,还应用了很多设计模式。绝对值得认真学习。
NHibernate ASP.NET 企业级应用案例。英文。网上找到的。案例所用的框架很明了也很经典,还应用了很多设计模式。绝对值得认真学习。
NHibernate is a port of Hibernate Core for Java to the .NET Framework. It handles persisting plain .NET objects to and from an underlying relational database. Given an XML description of your entities...
asp.net_spring_nhibernate_oracle 网上关于在.net开发的例子不多,这个例子能跑,遗憾的是,nhibernate在存储过程不完美,在oracle 传回多结果集,只能接受到第一个,官网上也是这样说的。 提示:开发前,先将...
有差Nhibernate spring.net的实例,可以运行
做培训做的一个简单框架 使用Nhibernate初学者可以参考一下 很干净,完整的代码包
NHibernate 是优秀的 ORM 框架,在开发中,我们希望能够集成使用这两个框架,在 Spring.NET 1.3.2 中对于当前的 NHibernate 3.2 提供了直接的支持。目前 NHibernate 已经发布了 3.3.1 ,但是在 Spring.NET 1.3.2 中...
再次向对NHibernate陌生的朋友介绍一下NHibernate,它是一个面向.NET 环境的对象/关系数据库映射工具。用来把对象映射到基于SQL 的关系模型数据结构中去。因此本书围绕这些主要讲解NHibernate是什么、如何建立第一个...