二十、自定义值类型

   开发者创建属于他们自己的值类型也是很容易的。比如说,你可能希望持久化Int64类型的属性, 持久化成为VARCHAR 字段。NHibernate没有内置这样一种类型。自定义类型能够映射一个属性(或集合元素)到不止一个数据库表字段。 比如说,你可能有这样的属性: Name { get; set; },这是String类型的,对应的持久化到三个字段:FIRST_NAMEINITIALSURNAME

要实现一个自定义类型,可以实现NHibernate.UserTypes.IUserTypeNHibernate.UserTypes.ICompositeUserType中的任一个, 并且使用类型的全限定类名来定义属性。请查看 NHibernate.DomainModel.DoubleStringType这个例子,看看它是怎么做的。

  1. <property name="TwoStrings" type="NHibernate.DomainModel.DoubleStringType, NHibernate.DomainModel">
  2. <column name="first_string"/>
  3. <column name="second_string"/>
  4. </property>

 注意使用<column>标签来把一个属性映射到多个字段的做法。

  ICompositeUserTypeIEnhancedUserTypeINullableUserTypeIUserCollectionType, 和 IUserVersionType接口为更特殊的使用方式提供支持。

  你甚至可以在一个映射文件中提供参数给一个IUserType。 为了这样做, 你的UserType必须实现NHibernate.UserTypes.IParameterizedType接口。为了给自定义类型提供参数,你可以在映射文件中使用<type>元素。

  1. <property name="priority">
  2. <type name="MyCompany.UserTypes.DefaultValueIntegerType">
  3. <param name="default">0</param>
  4. </type>
  5. </property>

  现在,IUserType 可以从传入的IDictionary对象中得到default 参数的值。

  尽管 NHibernate 内建的丰富的类型和对组件的支持意味着你可能很少 需要使用自定义类型。不过, 为那些在你的应用中经常出现的(非实体)类使用自定义类型也是一个好方法。例如, 一个MonetaryAmount类使用ICompositeUserType来映射是不错的选择,虽然他可以很容易地被映射成组件。这样做的动机之一是抽象。使用自定义类型,以后假若你改变表示金额的方法时,它可以保证映射文件不需要修改。

二十一、任意(Any)类型映射

   这是属性映射的又一种类型。<any> 映射元素定义了一种从多个表到类的多态关联。 这种类型的映射常常需要多于一个字段。第一个字段持有被关联实体的类型,其他的字段持有标识符。 对这种类型的关联来说,不可能指定一个外键约束,所以这当然不是映射(多态)关联的通常的方式。 你只应该在非常特殊的情况下使用它(比如,审计log,用户会话数据等等)。

  meta-type属性使得应用程序能指定一个将数据库字段的值映射到持久化类的自定义类型。 这个持久化类包含有用id-type指定的标识符属性。 你必须指定从meta-type的值到类名的映射。

  1. <any name="being" id-type="Int64" meta-type="string">
  2. <meta-value value="TBL_ANIMAL" class="Animal"/>
  3. <meta-value value="TBL_HUMAN" class="Human"/>
  4. <meta-value value="TBL_ALIEN" class="Alien"/>
  5. <column name="table_name"/>
  6. <column name="id"/>
  7. </any>

  NHibernate也支持meta-type="class"标签,这个例子里meta-value不是必须的, 因为 meta-value就是持久化类名(persistentClass.FullName)。

  1. <any name="being" id-type="Int64" meta-type="class">
  2. <column name="table_name"/>
  3. <column name="id"/>
  4. </any>

 但你使用meta-type="class"在查询语句里设置参数时,你必须使用下面的代码:

  1. SetParameter("paramName", typeof(YourClass).FullName, NHibernateUtil.ClassMetaType)

 映射文件部分:

  1. <any
  2. name="propertyName" (1)
  3. id-type="idtypename" (2)
  4. meta-type="metatypename" (3)
  5. cascade="cascade_style" (4)
  6. access="field|property|ClassName" (5)
  7. optimistic-lock="true|false" (6)
  8. >
  9. <meta-value ... />
  10. <meta-value ... />
  11. .....
  12. <column .... />
  13. <column .... />
  14. .....
  15. </any>

 说明:

(1)

name: 属性名。

(2)

id-type: 标识符类型。

(3)

meta-type (可选 -默认是 string): 允许辨别标志(discriminator)映射的任何类型。

(4)

cascade (可选 -默认是none): 级联的类型。

(5)

access (可选 -默认是 property): NHibernate 用来访问属性值的策略。

(6)

optimistic-lock (可选 -默认是 true): 表明更新此组件是否需要获取乐观锁。换句话说,当这个属性变脏时,是否增加版本号(Version) 。

二十二、SQL中引号包围的标识符

   你可强制NHibernate在生成的SQL中把标识符用引号前后包围起来,这需要在映射文档中使用反向引号(`)把表名或者字段名包围(可能比较拗口,请看下面的例子)。NHibernate会使用相应的SQLDialect(方言)来使用正确的引号风格(通常是双引号,但是在SQL Server中是括号,MySQL中是反向引号)。
  1. <class name="LineItem" table="`Line Item`">
  2. <id name="Id" column="`Item Id`"/><generator class="assigned"/></id>
  3. <property name="ItemNumber" column="`Item #`"/>
  4. ...
  5. </class>

二十三、模块化映射文件

   允许在独立的映射文档中定义subclassjoined-subclass,直接位于hibernate-mapping下。这就可以让你每次扩展你的类层次的时候,加入新的映射文件就行了。在子类的映射中你必须指定一个extends属性,指明先前已经映射过的超类。使用这个功能的时候,一定要注意映射文件的排序是非常重要的!
  1. <hibernate-mapping>
  2. <subclass name="Eg.Subclass.DomesticCat, Eg"
  3. extends="Eg.Cat, Eg" discriminator-value="D">
  4. <property name="name" type="string"/>
  5. </subclass>
  6. </hibernate-mapping>

二十四、数据库生成属性(Generated Properties)

   Generated properties指的是其值由数据库生成的属性。一般来说,如果对象有任何属性由数据库生成值,NHibernate应用程序需要进行刷新(Refresh)。但如果把属性标明为generated,就可以转由NHibernate来负责这个动作。实际上。对定义了generated properties的实体,每当NHibernate执行一条SQL INSERT或者UPDATE语句,会立刻执行一条select来获得生成的值。

  被标明为generated的属性还必须是 non-insertable和 non-updateable的。只有(version)(可选),时间戳 (可选)和属性可以被标明为generated。

  • never (默认) 标明此属性值不是从数据库中生成。
  • insert - 标明此属性值在insert的时候生成,但是不会在随后的update时重新生成。比如说创建日期就归属于这类。
  • always - 标明此属性值在insert和update时都会被生成。

二十五、数据库辅助对象

   帮助CREATE和DROP任意数据库对象,与NHibernate的schema交互工具组合起来,可以提供在NHibernate映射文件中完全定义用户schema的能力。虽然这是为创建和销毁trigger(触发器)或stored procedure(存储过程)等特别设计的,实际上任何可以在IDbCommand.ExecuteNonQuery()方法中执行的SQL命令都可以在此使用(比如ALTER, INSERT,等等)。本质上有两种模式来定义辅助数据库对象。

  第一种模式是在映射文件中显式声明CREATE和DROP命令:

  1. <nhibernate-mapping>
  2. ...
  3. <database-object>
  4. <create>CREATE TRIGGER my_trigger ...</create>
  5. <drop>DROP TRIGGER my_trigger</drop>
  6. </database-object>
  7. </nhibernate-mapping>

  第二种模式是提供一个类,这个类知道如何组织CREATE和DROP命令。这个特别类必须实现NHibernate.Mapping.IAuxiliaryDatabaseObject接口。

  1. <hibernate-mapping>
  2. ...
  3. <database-object>
  4. <definition class="MyTriggerDefinition, MyAssembly"/>
  5. </database-object>
  6. </hibernate-mapping>

 你也可以在配置文件里设置参数传给数据库对象。

  1. <hibernate-mapping>
  2. ...
  3. <database-object>
  4. <definition class="MyTriggerDefinition, MyAssembly">
  5. <param name="parameterName">parameterValue</param>
  6. </definition>
  7. </database-object>
  8. </hibernate-mapping>

NHibernate可以调用IAuxiliaryDatabaseObject.SetParameterValues方法接受dictionary参数。

  还有,这些数据库对象可以特别指定为仅在特定的方言中才使用。

  1. <hibernate-mapping>
  2. ...
  3. <database-object>
  4. <definition class="MyTriggerDefinition"/>
  5. <dialect-scope name="NHibernate.Dialect.Oracle9Dialect"/>
  6. <dialect-scope name="NHibernate.Dialect.OracleDialect"/>
  7. </database-object>
  8. </hibernate-mapping>

NHIBERNATE之映射文件配置说明(转载4)的更多相关文章

  1. NHibernate之映射文件配置说明

    NHibernate之映射文件配置说明 1. hibernate-mapping 这个元素包括以下可选的属性.schema属性,指明了这个映射所引用的表所在的schema名称.假若指定了这个属性, 表 ...

  2. NHibernate之映射文件配置说明(转载3)

    十二.组件(component), 动态组件(dynamic-component) <component>元素把子对象的一些元素与父类对应的表的一些字段映射起来. 然后组件可以定义它们自己 ...

  3. NHibernate之映射文件配置说明(转载1)

    源博客:http://www.cnblogs.com/kissdodog/archive/2013/02/21/2919886.html 1. hibernate-mapping 这个元素包括以下可选 ...

  4. [转]NHibernate之映射文件配置说明

    1. hibernate-mapping 这个元素包括以下可选的属性.schema属性,指明了这个映射所引用的表所在的schema名称.假若指定了这个属性, 表名会加上所指定的schema的名字扩展为 ...

  5. NHibernate之映射文件配置说明(转载2)

    六.鉴别器   在"一棵对象继承树对应一个表"的策略中,<discriminator>元素是必需的, 它定义了表的鉴别器字段. 鉴别器字段包含标志值,用于告知持久化层应 ...

  6. NHibernate初入门之映射文件配置说明(三)

    转载逆心http://www.cnblogs.com/kissdodog/archive/2013/02/21/2919886.html 1. hibernate-mapping 这个元素包括以下可选 ...

  7. NHibernate之映射文件配置说

    1. hibernate-mapping 这个元素包括以下可选的属性.schema属性,指明了这个映射所引用的表所在的schema名称.假若指定了这个属性, 表名会加上所指定的schema的名字扩展为 ...

  8. 使用代码辅助生成工具CodeSmith -- 生成NHibernate的映射文件

    首先下载CodeSmith工具:在百度云中,在CodeSmith文件夹中. 安装,使用激活工具激活. 然后下载NHibernate模板,也是在百度云中,在CodeSmith文件夹中. 之后直接点击NH ...

  9. 使用CodeSmith快速生成映射文件和映射类

    一 CodeSmith简介 本文以表自动生成NHibernate的映射文件和映射类的实例来说明一下本软件的使用方法. CodeSmith是一种基于模板的代码生成工具,其使用类似于ASP.NET的语法来 ...

随机推荐

  1. 关于<%@ include file=" " %>与<jsp:include page=""></jsp:include>中的那些问题?

    今天在使用<%@ include file=" " %>指令时,竟然在页面中不让使用?这是怎么回事:问题如下图: 顿时被这个问题给搞到了!!!突然想到在以前的 JSP ...

  2. oracle单行函数之通用函数

    NVL (a,b) --当a=null时,返回b,否则返回a NVL2 (a, b, c) -- 当a=null时,返回c,否则返回b NULLIF (expr1, expr2) --当a=b时,返回 ...

  3. android 手机信息获取

    1. adb已安装 2. adb shell getprop 此时已列出所有相关信息

  4. phpcms V9 修改生成静态文件路径/html

    在论坛看到部分用户反馈这个问题,要修改的其实是html_root的值,默认是"/html"如果要生成在网站根目录的话,这个值则要为空.论坛上现在看到的办法是打开caches\con ...

  5. Windows下安装Emacs+Sbcl+Slime

    前言 其实网上已经有很多类似的文章了,我也是按照上面的来做.在做的过程中会遇到几个很坑的地方,我自己也是折腾了好久才弄好.所以现在写出来希望能对大家有所帮助. 正文 下载和安装Emacs http:/ ...

  6. 『C # 开发』技能 Get√ ——制作CMD界面的简单GIF图片

    今天看到C#课本上个列子把星号(*)有规则打印在控制台中间位置 程序不难,利用的是光标定位函数Console.SetCursorPosition(x, y)做到的 心想是不是弄出一个动态的图案比较好玩 ...

  7. ural 1723 Sandro's Book

    http://acm.timus.ru/problem.aspx?space=1&num=1723 #include <cstdio> #include <cstring&g ...

  8. Printing Architecture

    Printing Architecture http://www.codeproject.com/Articles/8916/Printing-Architecture     This articl ...

  9. Android 使用Application总结

    Application 配置全局Context 第一步.写一个全局的单例模式的MyApplication继承自Application 覆盖onCreate ,在这个方法里面实例化Application ...

  10. 排序功能实现 jQuery实现排序 上移 下移

    效果 思路, 跟相邻元素,互换sort. 前提是每一个元素都有自己的sort值,不为零. <tr id="{sh:$vo.id}"> <td> <sp ...