假设有这样一个枚举:

/// <summary>
/// 字典项类型
/// </summary>
public enum DicItemType
{
[EnumDescription("程序使用")]
Program = , [EnumDescription("用户自定义")]
Custom =
}

NHibernate默认是映射为数据库中的数字类型,也就是0或者1。当我们使用数据库管理工具(例如PLSql/Developer)直接浏览数据库里的数据时,看到的一排又一排的0或者1,可读性不太好,总是要再去找枚举定义看0代表的是什么,1代表的又是什么。当使用存储过程写一些复杂查询时,也会写 CASE WHEN ITEM_TYPE = 1 THEN ... 或者 WHERE ITEM_TYPE = 1 这样包含了神奇数字的语句,可读性同样不好。虽然可以在后面跟上注释,也还是很烦。能不能让NHibernate把枚举保存为数据库里的字符串呢?例如不是保存0或者1,而是保存为 "Program" 或者 "Custom"。答案是肯定的。NHibernate提供了自定义类型的机制,通过将枚举映射为一个自定义类型,就可以自定义从数据库中读取枚举数据和保存枚举到数据库中的操作。下面详细描述一下实现方法。环境:.Net Framework 4.0,Oracle 11g R2,NHibernate 3.3.1.4000,FluentNHibernate 1.3.0.733,使用Oracle的64位ODP Oracle.DataAccess 4.112.3.0。

新增自定义类型

首先,需要新增一个自定义类型 NullableEnumCusType,它实现 IUserType 接口。这个自定义类型告诉NHibernate要把枚举保存为数据库中的 NHibernateUtil.AnsiString.SqlType 类型,以及如何从数据库中读取和保存枚举值(经由 NullSafeGet() 和 NullSafeSet() 方法)。

using System;
using NHibernate;
using NHibernate.SqlTypes;
using NHibernate.UserTypes; namespace Zen.Framework.Data
{
public class EnumCusType<TEnum> : IUserType where TEnum:struct
{
public object Assemble(object cached, object owner)
{
return DeepCopy(cached);
} public object DeepCopy(object value)
{
return value;
} public object Disassemble(object value)
{
return DeepCopy(value);
} public bool Equals(object x, object y)
{
return x.Equals(y); // 由于装箱后 x == y 会总是返回 false, 所以要使用 Equals()
} public int GetHashCode(object x)
{
return x.GetHashCode();
} public bool IsMutable
{
get { return true; }
} public object NullSafeGet(System.Data.IDataReader rs, string[] names, object owner)
{
return Enum.Parse(typeof(TEnum), NHibernate.NHibernateUtil.AnsiString.NullSafeGet(rs, names[]).ToString());
} public void NullSafeSet(System.Data.IDbCommand cmd, object value, int index)
{
NHibernate.NHibernateUtil.AnsiString.NullSafeSet(cmd, value.ToString(), index);
} public object Replace(object original, object target, object owner)
{
return target;
} public Type ReturnedType
{
get { return typeof(TEnum); }
} public NHibernate.SqlTypes.SqlType[] SqlTypes
{
get { return new SqlType[] { NHibernateUtil.AnsiString.SqlType }; }
}
}
}

EnumCusType

使用自定义类型

在实体的枚举属性映射时,要指定上面新增的自定义枚举类型。这样就已经可以实现枚举与数据库字符串之间的映射和转换了。

Map(t => t.ItemType, "ITEM_TYPE").CustomType<EnumCusType<DicItemType>>();

配置可为空类型的枚举

对于可为空类型的属性,例如:

/// <summary>
/// 字典项类型2
/// </summary>
public virtual DicItemType? ItemType2 { get; set; }

需要在增加一个针对可为空枚举的自定义类型:

using System;
using NHibernate;
using NHibernate.SqlTypes;
using NHibernate.UserTypes; namespace Zen.Framework.Data
{
public class NullableEnumCusType<TEnum> : IUserType
{
public object Assemble(object cached, object owner)
{
return DeepCopy(cached);
} public object DeepCopy(object value)
{
return value;
} public object Disassemble(object value)
{
return DeepCopy(value);
} public bool Equals(object x, object y)
{
if (x == null)
{
return x == y;
}
else
{
return x.Equals(y); // 由于装箱后 x == y 会总是返回 false, 所以要使用 Equals()
}
} public int GetHashCode(object x)
{
return x.GetHashCode();
} public bool IsMutable
{
get { return true; }
} public object NullSafeGet(System.Data.IDataReader rs, string[] names, object owner)
{
var v = NHibernate.NHibernateUtil.AnsiString.NullSafeGet(rs, names[]);
if (v == null)
{
return null;
}
else
{
return Enum.Parse(typeof(TEnum).GetGenericArguments()[], v.ToString());
}
} public void NullSafeSet(System.Data.IDbCommand cmd, object value, int index)
{
NHibernate.NHibernateUtil.AnsiString.NullSafeSet(cmd, (value == null? null : value.ToString()), index);
} public object Replace(object original, object target, object owner)
{
return target;
} public Type ReturnedType
{
get { return typeof(TEnum); }
} public NHibernate.SqlTypes.SqlType[] SqlTypes
{
get { return new SqlType[] { NHibernateUtil.AnsiString.SqlType }; }
}
}
}

NullableEnumCusType

然后把可为空类型的实体属性映射为可为空的自定义枚举类型,这样可为空的枚举也没问题了。

Map(t => t.ItemType2, "ITEM_TYPE2").CustomType<NullableEnumCusType<DicItemType>>();

使用 EnumConvention 统一配置自定义枚举类型

之前,我们是可以在 EnumConvention 中对枚举类型进行统一配置的,也就是说,在实体映射的时候,不需要区分枚举和普通的字符型以及数字型属性,例如可以这样:

/// <summary>
/// 字典类别
/// </summary>
public class DicItemEntityMap : InputItemMap<DicItem>
{
public DicItemEntityMap()
{
Table("SYS_DIC_ITEM");
Id(t => t.Id, "DIC_ITEM_ID");
Map(t => t.IsStop, "IS_STOP");
Map(t => t.ItemType, "ITEM_TYPE");
Map(t => t.ItemType2, "ITEM_TYPE2");
References(t => t.Category, "DIC_CATEGORY_ID");
}
}

这是因为 FluentNHibernate 提供了 Convention  机制可以对各种类型进行统一配置。例如只要实现这样一个 EnumConvention 就可以对所有枚举统一配置:

public class EnumConvention : IUserTypeConvention
{
public void Accept(FluentNHibernate.Conventions.AcceptanceCriteria.IAcceptanceCriteria<FluentNHibernate.Conventions.Inspections.IPropertyInspector> criteria)
{
// 匹配枚举或可为空枚举
criteria.Expect(x => x.Property.PropertyType.IsEnum
||
(x.Property.PropertyType.Name == "Nullable`1" &&
x.Property.PropertyType.GetGenericArguments().Length > &&
x.Property.PropertyType.GetGenericArguments()[].IsEnum));
} public void Apply(FluentNHibernate.Conventions.Instances.IPropertyInstance instance)
{
instance.CustomType(instance.Property.PropertyType);
}
}

EnumConvention

能不能把上面的 EnumConvention 改造一下,适配新增的 EnumCusType 和 NullableEnumCusType 呢?这样就不用每次都写 ” Map(t => t.ItemType, "ITEM_TYPE").CustomType<EnumCusType<DicItemType>>() “ 这样烦人的配置了。

但是你马上就会发现 EnumCusType 是非常坑爹的泛型,在 EnumConvention 里没法直接调用 instance.CustomType() 进行配置。好在微软提供了非常强大的Emit,动态生成一个泛型类型非常轻松:

public class EnumConvention : IUserTypeConvention
{
public void Accept(FluentNHibernate.Conventions.AcceptanceCriteria.IAcceptanceCriteria<FluentNHibernate.Conventions.Inspections.IPropertyInspector> criteria)
{
// 匹配枚举或可为空枚举
criteria.Expect(x => x.Property.PropertyType.IsEnum
||
(x.Property.PropertyType.Name == "Nullable`1" &&
x.Property.PropertyType.GetGenericArguments().Length > &&
x.Property.PropertyType.GetGenericArguments()[].IsEnum));
} public void Apply(FluentNHibernate.Conventions.Instances.IPropertyInstance instance)
{
//instance.CustomType(instance.Property.PropertyType);
if (instance.Property.PropertyType.Name == "Nullable`1")
{
// 转换为可空枚举自定义类型
Type t = TypeBuilder.GetType(
string.Format("Zen.Framework.Data.NullableEnumCusType`1[[{0}]]",
instance.Property.PropertyType.AssemblyQualifiedName)); instance.CustomType(t);
}
else
{
// 转换为非可空枚举自定义类型
Type t = TypeBuilder.GetType(
string.Format("Zen.Framework.Data.EnumCusType`1[[{0}]]",
instance.Property.PropertyType.AssemblyQualifiedName)); instance.CustomType(t);
}
}
}

使用Emit配置泛型自定义类型

配置NHibernate将枚举保存为Oracle数据库中的字符串的更多相关文章

  1. ||在oracle数据库中起到字符串拼接的作用

    例子:select org.id from org where inner_code like '12011601001' || '%' ||在oracle数据库中起到字符串拼接的作用,上面等同于'1 ...

  2. Oracle数据库中调用Java类开发存储过程、函数的方法

    Oracle数据库中调用Java类开发存储过程.函数的方法 时间:2014年12月24日  浏览:5538次 oracle数据库的开发非常灵活,不仅支持最基本的SQL,而且还提供了独有的PL/SQL, ...

  3. 详解大数据采集引擎之Sqoop&采集oracle数据库中的数据

    一.Sqoop的简介: Sqoop是一个数据采集引擎/数据交换引擎,采集关系型数据库(RDBMS)中的数据,主要用于在RDBMS与HDFS/Hive/HBase之间进行数据传递,可以通过sqoop i ...

  4. Oracle数据库中,误删除或者修改数据恢复方法

    在我们实际工作中,误删除或者修改Oracle数据库中的数据,怎么办呢?这里给大家分享一种解决办法.假如你误操作的时间不超过30分钟(数据库默认的回滚保持段里的数据时间,可以在pl/sql执行窗口按ct ...

  5. 将Oracle数据库中的数据写入Excel

    将Oracle数据库中的数据写入Excel 1.准备工作 Oracle数据库"TBYZB_FIELD_PRESSURE"表中数据如图: Excel模板(201512.xls): 2 ...

  6. --关于null在oracle数据库中是否参与计算,进行验证,

    --关于null在oracle数据库中是否参与计算,进行验证,with td as (select null id,1 name from dual ),td1 as ( select null id ...

  7. Oracle数据库中SYS、SYSTEM、DBSNMP、SYSMAN四用户的区别

    [转]   SYS.SYSTEM.DBSNMP. Oracle 数据库中 SYS.SYSTEM.DBSNMP.SYSMAN 四用户的区别 用户: SYS 用户: SYS,默认密码为 CHANGE_ON ...

  8. 【转】Oracle数据库中Sequence的用法

    在Oracle数据库中,sequence等同于序列号,每次取的时候sequence会自动增加,一般会作用于需要按序列号排序的地方. 1.Create Sequence (注释:你需要有CREATE S ...

  9. oracle数据库中提供的5种约束

    约束作用:用来保持数据的完整性,防止无效数据进入到数据库中.oracle数据库中提供的5种约束,都是限定某个列或者列的组合的.1.主键约束(PRIMARY KEY):在一个表中能唯一的标识一行.主键可 ...

随机推荐

  1. AVAssetReader+AVAssetReaderTrackOutput播放视频

    该文章引用自:http://www.jianshu.com/p/3d5ccbde0de1 IOS 微信聊天发送小视频的秘密(AVAssetReader+AVAssetReaderTrackOutput ...

  2. Android 谈谈封装那些事 --BaseActivity 和 BaseFragment(二)

      1.前言 昨天谈了BaseActivity的封装,Android谈谈封装那些事--BaseActivity和BaseFragment(一)有很多小伙伴提了很多建议,比如: 通用标题栏可以自定义Vi ...

  3. windows和linux之间“/”, "\\"的区别

    在windows下编程操作文件目录时,文件目录一般是如下的形式: C:\\folder1\\folder2\\folder3\\file.txt 而在Linux或者Mac系统下,文件目录则一般是这样子 ...

  4. 大家都在用PDA条码扫描枪管理企业仓库 PDA无线数据采集程序

    PDA数据采集器又称之为手持终端,这些都是用于扫描货物条码统计数据用的,PDA扫描枪有效提高企业仓库管理,在仓库管理中引入条码技术,对仓库的到货检验.入库.出库.调拨.移库移位.库存盘点等各个作业环节 ...

  5. [spring源码学习]二、IOC源码——配置文件读取

    一.环境准备 对于学习源码来讲,拿到一大堆的代码,脑袋里肯定是嗡嗡的,所以从代码实例进行跟踪调试未尝不是一种好的办法,此处,我们准备了一个小例子: package com.zjl; public cl ...

  6. centos 6.5 yum安装 mysql 5.6

    1. 查看系统里面有没有mysql 的repo   yum repolist all | grep mysql 2. 如果没有发现,则需要配置repo 注意,如果要使用5.7 或者其他任何版本,只能有 ...

  7. 转:Delphi 函数大全

    Delphi 函数大全 - xiucaiyao的专栏 - 博客频道 - CSDN.NEThttp://blog.csdn.net/xiucaiyao/article/details/4544039 名 ...

  8. LNMP虚拟机开发环境配置--vagrant+virtualbox+ubuntu14.04

    工作一直用的是别人打包好的虚拟机开发环境,感觉确实很酷.所以准备自己配个开发环境,为之后自己开发一些有趣的东西做准备. ok,开始~~~ 一.安装软件 vagrant和virtualbox 此处需注意 ...

  9. js转盘抽奖

    这个是很简易的转盘,只用了html,css,js 通过css产生一个转盘上的指针,用js动态改变css中的transparent改变指针的角度.再添加一个背景图片类似于奖项的转盘 <!DOCTY ...

  10. VIM 解决中文乱码

    $ vim ~/.vimrc 加入以下内容 set fileencodings=utf-8,ucs-bom,gb18030,gbk,gb2312,cp936 set termencoding=utf- ...