hibernate多对一 一对一 一对多 多对多的配置方法

本文详细介绍了Hibernate中各种关联映射的方式,包括一对一、一对多、多对一及多对多关联映射,并通过实例展示了如何进行配置。

 


hihernate

一对多关联映射(单向Classes----->Student

 

一对多关联映射利用了多对一关联映射原理

 

多对一关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是多指向一
一对多关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是一指向多

 

也就是说一对多和多对一的映射策略是一样的,只是站的角度不同

 

在一一端维护关系的缺点:
 *
如果将t_student表里的classesid字段设置为非空,则无法保存
 *
因为不是在student这一端维护关系,所以student不知道是哪个班的,
  
所以需要发出多余的update语句来更新关系

 

hihernate一对多关联映射(双向Classes<----->Student

 

一对多双向关联映射:
 *
在一一端的集合上使用<key>,在对方表中加入一个外键指向一一端
 *
在多一端采用<many-to-one>
 
注意:<key>标签指定的外键字段必须和<many-to-one>指定的外键字段一致,否则引用字段的错误
 
如果在一端维护一对多关联关系,hibernate会发出多余的udpate语句,所以我们一般在多
的一端来维护关联关系

 

关于inverse属性:
 inverse
主要用在一对多和多对多双向关联上,inverse可以被设置到集合标签<set>上,
 
默认inversefalse,所以我们可以从端和端维护关联关系,
 
如果设置成inversetrue,则我们只能从多一端来维护关联关系
 
 
注意:inverse属性,只影响数据的存储,也就是持久化
<!--[if !supportLineBreakNewLine]-->
<!--[endif]-->

 

inversecascade
 * inverse
是关联关系的控制方向
 * cascade
操作上的连锁反应

 

hibernate一对一主键关联映射(单向关联Person---->IdCard

 

一对一主键关联映射:让两个实体对象的id保持相同,这样可以避免多余的字段被创建

 

具体映射:

 

 <id name="id">
  <!-- person
的主键来源idCard,也就是共享idCard的主键 -->
  <generator class="foreign">
   <param name="property">idCard</param>
  </generator>
 </id>
 <property name="name"/>
 <!-- one-to-one
标签的含义,指示hibernate怎么加载它的关联对象,默认根据主键加载,
 constrained="true"
 表明当前主键上存在一个约束,person的主键作为外键参照了idCard 
  -->
 <one-to-one name="idCard" constrained="true"/>

 

hibernate一对一主键关联映射(双向关联Person<---->IdCard

 

需要在idcard映射文件中加入<one-to-one>标签指向person,指示hibernate如何加载person
默认根据主键加载

 

hibernate一对一唯一外键关联映射(单向关联Person---->IdCard

 

一对唯一外键关联映射是多对一关联映射的特例

 

可以采用<many-to-one>标签,指定多的一端的unique=true,这样就限制了多的一端的多重性为一
通过这种手段映射一对一唯一外键关联

 

hibernate一对一唯一外键关联映射(双向关联Person<---->IdCard

 

一对一唯一外键关联双向,需要在另一端(idcard),添加<one-to-one>标签,指示hibernate如何加载
其关联对象,默认根据主键加载person,外键关联映射中,因为两个实体采用的是person的外键维护的关系,
所以不能指定主键加载person,而要根据person的外键加载,所以采用如下映射方式:
<one-to-one name="person" property-ref="idCard"/>

 

hibernate多对一关联映射

 

关联映射的本质:
 *
将关联关系映射到数据库,所谓的关联关系是对象模型在内存中的一个或多个引用

 

<many-to-one>会在多的一端加入一个外键,指向一的一端,这个外键是由<many-to-one>
中的column属性定义的,如果忽略了这个属性那么默认的外键与实体的属性一致

 

<many-to-one>标签的定义示例:
  * <many-to-one name="group" column="groupid"/>
 
理解级联的含义?
 *
是对象的连锁操作 
hibernate
多对多关联映射(单向User---->Role)

 

具体映射方式:
 <set name="roles" table="t_user_role">
  <key column="userid"/>
  <many-to-many class="com.bjsxt.hibernate.Role" column="roleid"/>
 </set>
hibernate
多对多关联映射(双向User<---->Role)

 

映射方法:
  <set name="roles" table="t_user_role">
   <key column="userid"/>
   <many-to-many class="com.bjsxt.hibernate.Role" column="roleid"/>
  </set>
table
属性值必须和单向关联中的table属性值一致
<key>
column属性值要与单向关联中的<many-to-many>标签中的column属性值一致
<many-to-many>中的column属性值要与单向关联中<key>标签的column属性值一致

 

 


hibenate是一种优秀的对象持久化技术,可以实现关系型数据库和对象之间的映射,也提供了数据查询和恢复机制,可以大幅度减少开发时人工使用SQL和JDBC处理数据的时间.下面说说hibernate多对多关系的实现.

article表
CREATE TABLE `article` (
  `article_id` bigint(20) NOT NULL auto_increment,
  `chi_title` varchar(150) default NULL,
  `eng_title` varchar(150) default NULL
  PRIMARY KEY  (`article_id`),
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
 
article类

public class Article implements Serializable {
    private Long articleId;

    private String chiTitle;

    private String engTitle;
 
    private Set columns;

  ......

}

article配置文件:

<class
    name="com.nmce.web.model.Article"
    table="article"
    lazy="false"
>

    <id
        name="articleId"
        type="java.lang.Long"
        column="article_id"
    >
   
        <generator class="increment" />
    </id>

    <property
        name="chiTitle"
        type="java.lang.String"
        column="chi_title"
        length="150"
    />
    <property
        name="engTitle"
        type="java.lang.String"
        column="eng_title"
        length="150"
    />

      <set name="columns"
             table="col_article"
             cascade="save-update">            
             <key column="article_id"/>
            
         <many-to-many class="com.nmce.web.model.Column"
                           column="column_id"/>
       </set>
   

</class>

 

ncolumn表:

CREATE TABLE `ncolumn` (
  `column_id` int(11) NOT NULL auto_increment,
  `chi_title` varchar(100) NOT NULL,
  `eng_title` varchar(100) NOT NULL,
  PRIMARY KEY  (`column_id`),
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

 

ncolumn类:

public class Column implements Serializable {

    private Integer columnId;
    private String chiTitle;

    private String engTitle;

    private Set articles;

    ......

}

ncolumn配置文件:

<class
    name="com.nmce.web.model.Column"
    table="ncolumn"
    lazy="false"
>

    <id
        name="columnId"
        type="java.lang.Integer"
        column="column_id"
    >
   
        <generator class="increment" />
    </id>

    <property
        name="chiTitle"
        type="java.lang.String"
        column="chi_title"
        not-null="true"
        length="100"
    />
    <property
        name="engTitle"
        type="java.lang.String"
        column="eng_title"
        not-null="true"
        length="100"
    />
    
    <set name="articles"
             table="col_article"
             cascade="save-update">            
             <key column="column_id"/>
             <many-to-many class="com.nmce.web.model.Article"
                           column="article_id"/>
       </set>

</class>

 

关联表col_article

CREATE TABLE `col_article` (
  `article_id` bigint(20) NOT NULL default '0',
  `column_id` int(11) NOT NULL default '0',
  PRIMARY KEY  (`article_id`,`column_id`),
  KEY `columnFK1` (`column_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

关联表col_article不需要配置文件和PK类

 

实现插入:

ArticleForm af = (ArticleForm)form;

Article a = new Article();
String[] selectct = req.getParameterValues("selectct");

BeanUtils.copyProperties(a,af);

Set s = new HashSet();
  for(int i=0;i<selectct.length;i++){
   Column c = this.columnService.getColumnById(new Integer(selectct[i]));
   s.add(c);
  }

 a.setColumns(s);
  boolean b = this.articleService.addArticle(a);

就可以实现在添加文章的同时插入关联表

 

 


 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值