您的位置:知识库 » .NET技术

NHibernate3剖析:Mapping篇之ConfORM实战(2):原理

作者: 李永京  来源: 博客园  发布时间: 2010-09-19 23:25  阅读: 1448 次  推荐: 0   原文链接   [收藏]  

  ConfORM概述

  在上一节中,我用一个简单的例子描述了ConfORM简单使用。留下了很多疑问,大家不解为何使用ConfORM以及怎么使用ConfORM,其内部原理是什么。这节,我们先注重了解一些ConfORM的原理。

  你可以到http://code.google.com/p/codeconform/ 获取ConfORM

  ConfORM重要接口

  ConfORM的核心就是实例化一个ObjectRelationalMapper对象和Mapper对象,配置Domain对象,调用Mapper对象的CompileMappingFor()方法生成HbmMapping。即上一节中我们所写的:

//Code Snippets Copyright http://lyj.cnblogs.com/
public static HbmMapping GetMapping()
{
//初始化ObjectRelationalMapper类
var orm = new ObjectRelationalMapper();
//配置Domain为TablePerClass
orm.TablePerClass<Domain>();
//在这里可以调用ObjectRelationalMapper类一些方法配置Domain语义
//使用orm参数初始化Mapper类
var mapper = new Mapper(orm);
//在这里可以调用Mapper类一些方法配置Domain的Mapping
//调用Mapper类的CompileMappingFor方法编译生成HbmMapping对象
return mapper.CompileMappingFor(new[] { typeof(Domain) });
}

  在了解这段代码之前,先看看ConfORM的重要接口:

  IDomainInspector接口

  IDomainInspector接口用来描述我们的领域模型,按照ORM术语定义。是ConfORM的切入点,同时也是Mapping类的驱动。由ObjectRelationalMapper类实现这个接口。

IDomainInspector  IObjectRelationalMapper接口

  首先回顾下一些ORM术语:

  面向对象的三种继承策略:

  • TablePerClass:每个类一张表映射策略
  • TablePerClassHierarchy:每个类分层结构一张表映射策略
  • TablePerConcreteClass:每个具体类一张表映射策略

  在Entity模型属性中,主要有主键属性、持久化属性、非持久化属性。IObjectRelationalMapper接口使用ORM术语来描述Domain模型,包含了三种继承策略和Entity模型的各种属性设置方法:

  • 主键(Poid):每个实体都有自己的状态和生命周期,在数据库中的记录需要一个主键来识别。
  • 持久化属性:一般有基本属性,还有版本号属性(VersionProperty)、NaturalId属性、各种集合属性(Set、Bag、List、Array、Dictionary、Complex、HeterogeneousAssociation)、各种关联关系属性(ManyToMany、ManyToOne、OneToOne)。
  • 非持久化属性:不是所有的Domain、属性都需要做持久化,当我们不需要映射类或者属性时可以使用Exclude、ExcludeProperty排除。

  另外附加一个关联集合的级联方法:Cascade

IObjectRelationalMapper  IExplicitDeclarationsHolder接口

  显式声明Domain语义接口,使用IObjectRelationalMapper接口方法显式声明自定义Domain语义,ConfORM会添加到相应集合中。

IExplicitDeclarationsHolder  IPatternsHolder接口

  用于隐式声明Domain语义接口,主要原理是其默认实现类(DefaultNHibernatePatternsHolder类)中在各个集合里默认定义了相匹配的模式供我们来匹配其成员。

IPatternsHolder  例如:实体主键Poid默认使用PoIdPattern:如果实体成员的名称是"id"或者"poid"或者"oid"(不区分大小写)的话,就认为是实体的主键。

  ObjectRelationalMapper类

  ObjectRelationalMapper类实现IDomainInspector接口和IObjectRelationalMapper接口。IObjectRelationalMapper接口实现方法:用于把自定义的Domain语义配置到IExplicitDeclarationsHolder接口的相应集合中。IDomainInspector接口实现方法:用于验证Domain语义,它从两种角度去验证Domain语义:

  • 其一是在IExplicitDeclarationsHolder接口默认实现(ExplicitDeclarationsHolder类)中相关集合中显式匹配。
  • 其二是在IPatternsHolder接口的默认实现(DefaultNHibernatePatternsHolder类)中相关集合中隐式匹配。

  Mapping类

  Mapping类是ObjectRelationalMapper和NHibernate映射的桥梁。在ConfOrm.NH命名空间下,Mapping类通过IDomainInspector接口来分析Domain模型语义,我们通过CompileMappingFor()方法或者CompileMappingForEach()方法根据这些语义把程序中的Domain模型编译并转换为NHibernate使用的HbmMapping对象。

Mapper  ConfORM重要模式

  Mapping类除了CompileMappingFor()、CompileMappingForEach()方法之外,还为我们引入了三个重要模式,分别为:模式适配器(pattern-applier)模式、通用定制化(generic-customizer)模式、特定定制化(specific-customizer)模式。

  模式适配器(pattern-applier)

  模式适配器(pattern-applier),顾名思义,就是Domain模型按模式匹配,如果符合这个模式就进行相应操作。ConfORM默认提供了很多pattern-applier(即DefaultPatternsAppliersHolder类)。我们可以通过Mapping类的PatternsAppliers属性查看。也可以通过AddXXXPattern()方法增加模式适配器(pattern-applier)。

  通用定制化(generic-customizer)

  通用定制化(generic-customizer),就是指对通用的类型实现定制化。 我们不需要准确的知道这个类型的真实映射。如果要定制化一个类,我们不需要知道这个类最终被映射为class,subclass,joined-subclass,union-subclass还是component,只是把这个类设置一些属性。同理,你定制化一个集合不需要知道这个集合将映射为bag,set,array,list还是map(dictionary)。通用定制化由Mapping类的Customize()方法提供。

  特定定制化(specific-customizer)

  特定定制化(specific-customizer),就是对特定的类或者集合实现定制化。与通用定制化恰恰相反,我们对特定的类或者特定的某个集合设置一些定制化属性,这个定制化仅对当前你定制的对象有用。

特定定制化由Mapping类的Class()、Subclass()、JoinedSubclass()、UnionSubclass()、Component()方法提供。

  结语

  这篇文章了解一些ConfORM的原理,以后的文章都是以这篇文章为基础展示ConfORM各种应用。

  参考资料

  Fabio Maulo:ConfORM: NHibernate un-Mapping

0
0
标签:NHibernate

.NET技术热门文章

    .NET技术最新文章

      最新新闻

        热门新闻