zoukankan      html  css  js  c++  java
  • [转]NHibernate之旅(8):巧用组件之依赖对象

    本节内容

    • 引入
    • 方案1:直接添加
    • 方案2:巧用组件
    • 实例分析
    • 结语

    引入

    通过前面7篇的学习,有点乏味了~~~这篇来学习一个技巧,大家一起想想如果我要在Customer类中实现一个Fullname属性(就是Firstname和Lastname的组合)该怎么做呢?

    方案1:直接添加

    “我知道!修改Customer类,添加一个Fullname属性!即Customer.Fullname!”

    “恩,完全正确......”

    “这就意味着在Customer类中把Firstname和Lastname两个属性重新修改组合为Fullname属性。这样的话,如果有其它的类(像Vendor、Shiper)使用了Firstname和Lastname两个属性,这就需要修改很多业务逻辑。那你的麻烦可就大了,还有什么方法吗?”

    “.........”

    方案2:巧用组件

    NHibernate中,提供了组件(Component)和动态组件来帮助我们完成这件事情。其实组件在NHibernate中为了不同目的被重复使用。这里我们使用它来依赖对象

    映射文件中,<component>元素把子对象的一些属性映射为父类对应的表的一些字段。然后,组件可以定义它们自己的属性、组件或者集合。

    下面用两幅图显示组件和动态组件两个节点映射属性:

    节点映射属性

    看看这些映射属性:

    • access(默认property):NHibernate用来访问属性的策略
    • class(默认通过反射得到的属性类型):组件(子)类的名字
    • insert:被映射的字段是否出现在SQL的INSERT语句中
    • name:属性名propertyName
    • update:被映射的字段是否出现在SQL的UPDATE语句中
    • <property>子元素:为组件(子)类的一些属性与表字段之间建立映射
    • <parent>子元素:在组件类内部就可以有一个指向其容器的实体的反向引用

    <dynamic-component>元素允许一个IDictionary作为组件映射,其中属性名对应字典中的键。这又是使用组件的另一种用法。

    知道上面的知识,我们该想想上面的问题该如何利用组件来实现了吧。

    实例分析

    我们用一幅图来展示我们这节所说的一切:

    使用组件实例演示

    开始动手吧!

    1.新建Name类

    namespace DomainModel.Entities
    {
        public class Name
        {
            public string Firstname { get; set; }
            public string Lastname { get; set; }
            public string Fullname
            {
                get
                {
                    return Firstname + " " + Lastname;
                }
            }
        }
    }

    简单的说,这个类用于组合Fullname属性。

    2.修改Customer类

    namespace DomainModel.Entities
    {
        public class Customer
        {
            public virtual int CustomerId { get; set; }
            public virtual int Version { get; set; }
            public virtual Name Name { get; set; }
        }
    }

    修改Customer类,去除原来的Firstname和Lastname属性,添加Name属性。这时Name作为Customer的一个组成部分。需要注意的是:和原来Firstname和Lastname属性一样,需要对Name的持久化属性定义getter和setter方法,但不需要实现任何的接口或声明标识符字段。

    3.修改Customer映射

    <?xml version="1.0" encoding="utf-8" ?>
    <hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
                               assembly="DomainModel" namespace="DomainModel">
      <class name ="DomainModel.Entities.Customer,DomainModel" table="Customer">
        <id name="CustomerId" column="CustomerId" type="Int32" unsaved-value="0">
          <generator class ="native"></generator>
        </id>
        <version name="Version" column="Version" type="integer" unsaved-value="0"/>
        <component name="Name" class="DomainModel.Entities.Name,DomainModel">
            <property name="Firstname" column ="Firstname" type="string"
                                 length="50" not-null="false" unique-key="UC_CustomerName"/>
            <property name ="Lastname" column="Lastname" type="string"
                                 length="50" not-null="false" unique-key="UC_CustomerName"/>
        </component>
      </class>
    </hibernate-mapping>

    首先定义Component的一些属性,指定属性名和组件映射的类名。再使用<property>子元素,为Name类的Firstname、Lastname属性与表字段之间建立映射。是不是很简单~~

    这时Customer表中还是CustomerId、Version、Firstname、Lastname字段。完全不需要修改数据库表结构哦。

    这里需要注意两点:

    1. 就像所有的值类型一样,组件不支持共享引用。组件的值为空从语义学上来讲是专有的。每当重新加载一个包含组件的对象,如果组件的所有字段为空,那么NHibernate将假定整个组件为空。对于绝大多数目的,这样假定是没有问题的。
    2. 组件的属性可以是NHibernate类型(包括集合、多对一关联以及其它组件)。嵌套组件不应该作为特殊的应用被考虑。NHibernate趋向于支持设计细粒度的对象模型。

    4.编写方法

    这时,我们需要修改或者重新编写新的方法来实现我们想要的逻辑。

    public IList<Customer> ReturnFullName(string firstname, string lastname)
    {
        return _session
             .CreateQuery("select from Customer c where c.Name.Firstname=:fn and c.Name.Lastname=:ln")
            .SetString("fn", firstname)
            .SetString("ln", lastname)
            .List<Customer>();
    }

    现在,我们访问Customer的Firstname、Lastname属性,只需要在原来的基础上通过Name访问,例如上面修改的情况,看看上面图片上怎么访问的吧,一目了然。

    如果我们要添加一个Customer怎么办呢?代码片段如下所示:

    var customer = new Customer() { Name = new Name() { Firstname = "YJing", Lastname = "Lee" } };

    5.测试方法

    有了上面的方法,我们编写一个测试用例测试一下这个方法吧:看看结果测试成功,OK。

    [Test]
    public void ReturnFullNameTest()
    {
        IList<Customer> customers = _relation.ReturnFullName("YJing","Lee");
        foreach (Customer c in customers)
        {
            Assert.AreEqual("YJing Lee", c.Name.Fullname);
        }
    }

    结语

    这一篇像大家介绍一个使用组件技巧,通过组件可以改善我们的对象模型,而数据库结构不需要变化。通过这一篇的技巧,利用组件来映射来依赖对象,可以非常连贯的引入NHibernate中的多表映射关系、集合等内容,这些才是NHibernate中的亮点,就连LINQ都比不过它。从下篇开始就来学习NHibernate中的闪光点。

  • 相关阅读:
    jsp与spring mvc后台controller间参数传递处理之总结
    又一个无效的列类型错误Error setting null for parameter #7 with JdbcType NULL . Try setting a different JdbcType for this parameter or a different jdbcTypeForNull configuration property. Cause: java.sql.SQLExcept
    SSM文件上传要点总结
    关于SSM中mybatis向oracle添加语句采用序列自增的问题
    oracle和mysql的一些区别
    mapper.xml实现oracle的分页语句
    2.数组的解构赋值
    3.Vue 实例
    2.Vue.js 是什么
    1. vue 的安装
  • 原文地址:https://www.cnblogs.com/siyunianhua/p/3984604.html
Copyright © 2011-2022 走看看