首页 | 互联网 | IT动态 | Cisco | Windows | Linux | Java | .Net | Oracle | 华为 | 存储世界 | 服务器 | 网络设备 | IDC | 安全 | 求职招聘
IT培训 | 数字网校 | 技术专题 | 电子书下载 | 教学视频 | 网页设计 | 平面设计 | 解决方案 | 直播室 | 虚拟考场 | 搜索 | 博客 | 沙龙 | 论坛
中国IT实验室Linux频道
中国IT教育
 
首页 资讯动态 认证考试 新手入门 核心技术 高级技术 J2EE J2ME Java&XML 开源技术 其他技术 RSS订阅 论坛 专题
您现在的位置: 中国IT实验室 >> Java >> 开源技术 >> Hibernate >> 文章正文

Hibernate的映射关联关系

文章来源中国IT实验室收集整理 作者佚名 更新时间2008-5-7 保存本文保存本文 推荐给好友推荐给好友 收藏本页收藏本页

    Hibernate的映射关联关系和我们现实世界里事物的关联关系一样.比如在UML语言中,以客户Customer和订单Order的关系为例.一个客户可以发送多个订单,而一个订单只能属于一个客户,这是一对多的关联,因此可以成为单向关联.如果同时包含了两两种关联关系,就成为双向关联.在关系数据库中只有外键参照主键的关系.所以关系数据库实际上至支持一对一,或一对多的单向关系.在类于类之间的关系中.要算多对一关系和数据库中的外键参照主键关系最匹配了.因此如果使用单向关联从订单到客户的多对一单向关联,在订单类中就要定义一个客户的属性.表示这个订单属于哪个客户,而客户类就无需定义存放订单的集合属性了.下面写一个简单的例子.
    //首先定义客户类
    public class Customer implements Sreializable {
    private Long id;
    private String name;
    //省略属性的访问方法
    }
    //然后定义订单类
    public class Order implements Sreializable {
    private Long id;
    private String orderName;
    private Customer customer;
    //省略属性的访问方法,要注意的是Customer的访问方法.
    }
    Customer类的所有属性和CUSTOMERS表的所有属性一一对应,创建起来就比较简单了.下面主要看一下Order类的映射文件.
    <property name=\"orderName\" type=\"string\">
    <column name=\"ORDER_NAME\" length=\"15\"/>
    </property>
    因为customer属性是是Customer类型,而ORDERS表的CUSTOMER_ID是整数类型,是不匹配的.所以我们不能用普通的<property>元素来定义,而我们需要使用<many-to-one>元素来配置了.
    <many-to-one name=\"customer\" column=\"CUSTOMER_ID\" class=\"包名.Customer\" not-null=\"true\"/>
    <many-to-one>元素负责建立Order订单类的customer属性和数据库中的CUSTOMER_ID外键字段之间的映射.
    name:设定映射文件的属性名
    column:设定和持久化类对应的表的外键名
    class:设定持久化类的属性的类型,这里指定具体的类,也就是主键存在的类
    not-null:设定为true表示customer属性不允许为null,默认是false,这个属性会影响到bhm2ddl工具,会为ORDERS表的CUSTOMER_ID外键设置为不允许空的约束,但是不会影响到hbm2java工具生长java源代码.此外还会影响到Hibernate运行时的行为,在保存Order对象的时候会检查customer属性是否为null.用hbm2ddl编译之后得到的数据库文件如下:

    create table CUSTOMERS (
    ID bigint not null,
    NAME varchar(15),
    primary key (ID)
    );
    create table ORDERS (
    ID bigint not null,
    ORDER_NUMBER varchar(15),
    CUSTOMER_ID bigint not null,
    primary key (ID)
    );
    alter table ORDERS add index FK8B7256E516B4891C (CUSTOMER_ID),
    add constraint FK8B7256E516B4891C foreign key (CUSTOMER_ID) references CUSTOMERS (ID);
    看到结果我们可以简单的把<many-to-one>理解为在数据库中,创建外键的作用.上边这个例子就简单的演示了Hibernate映射的一对一关联关系,至于一对多的关联关系比这个稍微复杂一点.而且可以看出,当Hibernate持久化一个临时对象的时候,在默认的情况下它不会自动持久化关联其他临时对象,而是会抛出TransientObjectException异常.如果希望Hibernate持久化对象的时候也自动持久化说关联的对象,就要把<many-to-one>元素的cascade属性设置为save-update,表示级联操作的意思,cascade属性的默认值为none.当这个属性设置OK了.数据库就实现了级联保存更新的操作.
      在类和类之间建好了关联关系之后,就可以方便的从一个对象得到它关联的对象.例如Customer customer=order.getCustomer();这样获得的了Customer对象了.但是如果想获得所有属于Customer客户的Order订单对象,就涉及到了一对多双向关联了.在内存中,从一个对象导航都另一个对象要比从数据库中通过一个字段查询另一个字段快的多的多,但是也给编程的时候带来了麻烦,随意修改一个对象就可能牵一发而动全身,所以说双向的关联比较复杂,但是类和类之间到底建立单向还是双向关联,这个要根据业务需求来决定.比如说业务需求根据指定客户查询客户所有订单,根据指定的订单,查询出发这个订单的客户.这个时候我们不妨用多对一双向关联处理.其实上边的例子的映射文件已经简历了客户和订单之间的一对多双向关联关系,只不过要在客户类中加一个集合的属性: [Page]
    private set orders = new HashSet();
    public set getOrders() {
    return orders;
    }
    public void setOrders(Set orders) {
    this.orders = orders;
    }
    有了orders属性,客户就可以通过getOrders()方法或者客户的全部订单了,Hibernate在定义这个集合属性的时候必须声明为接口类型,但是不光光是Set还有Map和List,这样可以提高程序的强壮性,就是说set方法接受的对象只要是实现了Set接口就OK.避免出现null值的现象.这里要注意的是hbm2java工具生成类的集合属性的代码时,不会给它初始化一个集合对象的实例,这里我们需要自己手动修改,当然不修改也是可以的.接下来还要在customer.hbm.xml映射文件里映射集合类型的orders属性,当然这个和order表的的<many-to-one>同理,所以不能通过普通的<property>元素来设置属性和字段的映射关系.要使用<set>元素来设置:

    <set name=\"orders\" cascade=\"save-update\">
    <key column=\"CUSTOMER_ID\">
    <one-to-many class=\"包名.Order\">
    </set>
    name:设定类的属性名
    cascade:设置为save-update表示级联保存更新,当保存或更新Customer类的时候会级联保存更新跟它关联的Order类.
    <key>元素是用来设定跟持久化类关联的类的外键
    <one-to-many>元素看起来很熟悉,哦是设置外键的元素反过来了.这里它是用来设置所关联的持久化类的.这里设置为和客户关联的订单Order类,这里表明这个属性里要存放一组Order类型的对象.
    这个<set>元素是表示orders属性声明为set类型.
    <set>元素还有一个inverse属性,这个方法主要是在给已存在数据库中的字段建立关联的时候很有用.就是说当我们获得数据库中的两个表的两条记录的对象customer客户对象和order订单对象(映射文件已经建立了他们类和类之间的关联,但外键的值为null的情况下)然后我们想建立这个客户对象和订单对象之间的关联,我们要先调用order.setCustomer(customer);然后在调用customer.getOrder().add(order);在Hibernate自动清理缓存的持久化对象的时候会提交两条SQL语句.进行了两个update操作.但是实际上只修改了一条记录.重复的执行SQL语句是会降低系统的运行效率的,当把inverse属性设置为true的时候,同样的操作就会合并到一条SQL语句执行了,inverse默认为false;

[1] [2] 下一页  

【责编:Ken】

中国IT教育

相关产品和培训
文章评论
 友情推荐链接
 认证培训
 专题推荐

 ·关于Java框架技术专题
 ·XML全攻略技术专题
 ·JAVA开源技术介绍专题
 ·Java嵌入式开发之J2ME技术专题
 ·超前体验 Oracle 11g的5个新特性
 ·揭密使用VB.NET的五个实用技巧
 ·Oracle和SQL Server常用函数对比专题
 ·展现C#世界 C#程序设计专题
 ·Java入门 Tomcat的配置技巧精华专题
 ·Oracle RMAN物理备份技术详解
 今日更新
 社区讨论
 博客论点
 频道精选
 Java 频道导航