hibernate对于数据库的操作,全部利用面向对象的思维来理解和实现的。一般的单独表的映射,相信大家都没有问题,但是对于一些表之间的特殊关系,Hibernate提供了一些独特的方式去简化它。

今天就来说说多对一的关联映射。

数据库中有多对一的关系,Hibernate自然也有对象的多对一的关联关系。比如用户和用户组,一个用户只属于一个组,一个组有多名用户。我们就可以说用户和用户组的关系就是多对一的关系。用对象的uml图表示一下:

在Hibernate中如何来实现呢?首先定义这两个实体类:

  1. package com.bjpowernode.hibernate;
  2. /**
  3. * 用户组
  4. * @author Longxuan
  5. *
  6. */
  7. public class Group {
  8. private int id;
  9. private String name;
  10. public int getId() {
  11. return id;
  12. }
  13. public void setId(int id) {
  14. this.id = id;
  15. }
  16. public String getName() {
  17. return name;
  18. }
  19. public void setName(String name) {
  20. this.name = name;
  21. }
  22. }
  23. package com.bjpowernode.hibernate;
  24. /**
  25. * 用户类
  26. * @author Longxuan
  27. *
  28. */
  29. public class User {
  30. private int  id;
  31. private String name;
  32. private Group group;
  33. public Group getGroup() {
  34. return group;
  35. }
  36. public void setGroup(Group group) {
  37. this.group = group;
  38. }
  39. public int getId() {
  40. return id;
  41. }
  42. public void setId(int id) {
  43. this.id = id;
  44. }
  45. public String getName() {
  46. return name;
  47. }
  48. public void setName(String name) {
  49. this.name = name;
  50. }
  51. }

hibernate.cfg.xml配置文件:

  1. <!DOCTYPE hibernate-configuration PUBLIC
  2. "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
  3. "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
  4. <hibernate-configuration>
  5. <session-factory name="foo">
  6. <property name="hibernate.dialect" >org.hibernate.dialect.MySQLDialect</property>
  7. <property name="hibernate.show_sql">true</property><!-- 设置是否显示生成sql语句 -->
  8. <property name="hibernate.format_sql">false</property><!-- 设置是否格式化sql语句-->
  9. <mapping resource="com/bjpowernode/hibernate/Tables.hbm.xml" />
  10. </session-factory>
  11. </hibernate-configuration>

hibernate.properties配置文件:

  1. ## MySQL
  2. hibernate.dialect org.hibernate.dialect.MySQLDialect
  3. #hibernate.dialect org.hibernate.dialect.MySQLInnoDBDialect
  4. #hibernate.dialect org.hibernate.dialect.MySQLMyISAMDialect
  5. hibernate.connection.driver_class com.mysql.jdbc.Driver
  6. hibernate.connection.url jdbc:mysql://localhost:3306/hibernate_many2one
  7. hibernate.connection.username root
  8. hibernate.connection.password root

这是最基础的准备工作,多对一映射在对数据进行更改时,会有一些限制。当没有该用户组时,添加用户,就会出错,当该用户组有用户时,删除该用户组也会报错。

我们当然可以按一般的方法来做。添加的时候,先手动添加组,再添加用户。删除时,先删除所有的用户,再删除用户组。但是Hibernate为我们提供了一种便捷的方式——many-to-one。在映射文件hbm.xml中配置后,就可以不用再想那些限制了。Hibernate会自动添加上所引用的数据。

给出映射文件:

  1. <?xml version="1.0"?>
  2. <!DOCTYPE hibernate-mapping PUBLIC
  3. "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
  4. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
  5. <hibernate-mapping>
  6. <class name="com.bjpowernode.hibernate.User" table="t_user">
  7. <id name="id">
  8. <generator class="native" />
  9. </id>
  10. <property name="name"></property>
  11. <many-to-one name="group" column="groupid" cascade="save-update"></many-to-one>
  12. </class>
  13. <class name="com.bjpowernode.hibernate.Group" table="t_group">
  14. <id name="id">
  15. <generator class="native" />
  16. </id>
  17. <property name="name"></property>
  18. </class>
  19. </hibernate-mapping>

配置了many-to-one会自动在t_user表中创建外键groupid,与t_group的id映射。

many-to-one标签用到了cascade,指定两个对象之间的操作联动关系,对一个对象执行了操作之后,对其指定的级联对象也需要执行相同的操作。其属性值如下:

  • all:在所有的情况下都执行级联操作;
  • none:在所有情况下都不执行级联操作;
  • save-update:在保存和更新的时候执行级联操作;、
  • delete:在删除的时候执行级联操作。

测试类Many2OneTest:

  1. package com.bjpowernode.hibernate;
  2. import junit.framework.TestCase;
  3. import org.hibernate.Session;
  4. public class Many2OneTest extends TestCase {
  5. /**
  6. * 测试添加用户
  7. */
  8. public void testSave3(){
  9. Session session = null;
  10. try{
  11. session = HibernateUtils.getSession();
  12. session.beginTransaction();
  13. Group group = new Group();
  14. group.setName("提高班");
  15. User user1 = new User();
  16. user1.setName("张三");
  17. user1.setGroup(group);
  18. User user2 = new User();
  19. user2.setName("李四");
  20. user2.setGroup(group);
  21. //普通方法 :必须先保存group,再保存user
  22. //配置了many-to-one 则不用再手动save group了。
  23. //session.save(group);
  24. session.save(user1);
  25. session.save(user2);
  26. session.getTransaction().commit();
  27. }catch(Exception e){
  28. e.printStackTrace();
  29. session.getTransaction().rollback();
  30. }finally{
  31. HibernateUtils.closeSession(session);
  32. }
  33. }
  34. }

结果图:

执行测试前:  , 执行测试后: 

用many-to-one进行配置后,hibernate会自动去添加外键,而我们做的任何操作都不需要去考虑它的结构,也不用手动去维护这个关系,关系由Hibernate自动维护。这就是Hibernate的魅力所在。

一口一口吃掉Hibernate(四)——多对一单向关联映射的更多相关文章

  1. 【SSH系列】Hibernate映射-- 多对一单向关联映射

    在hibernate中非常重要的就是映射,在前面的博文中,小编简单的介绍了基本映射,基本映射是对一个实体进行映射,关联映射就是处理多个实体之间的关系,将关联关系映射到数据库中,所谓的关联关系在对象模型 ...

  2. Hibernate从入门到精通(十)多对多单向关联映射

    上一篇文章Hibernate从入门到精通(九)一对多双向关联映射中我们讲解了一下关于一对多关联映射的相关内容,这次我们继续多对多单向关联映射. 多对多单向关联映射 在讲解多对多单向关联映射之前,首先看 ...

  3. Hibernate从入门到精通(七)多对一单向关联映射

    上次的博文Hibernate从入门到精通(六)一对一双向关联映射中我们介绍了一下一对一双向关联映射,本次博文我们讲解一下多对一关联映射 多对一单向关联映射 多对一关联映射与一对一关联映射类似,只是在多 ...

  4. Hibernate(十)多对多单向关联映射

    上一篇文章Hibernate从入门到精通(九)一对多双向关联映射中我们讲解了一下关于一对多关联映射的 相关内容,这次我们继续多对多单向关联映射. 多对多单向关联映射 在讲解多对多单向关联映 射之前,首 ...

  5. Hibernate(七)多对一单向关联映射

    上次的博文Hibernate从入门到精通(六)一对一双向关联映射中我们介绍了一下一对一双向关联映射,本 次博文我们讲解一下多对一关联映射 多对一单向关联映射 多对一关联映射与一对一关联映射类 似,只是 ...

  6. [置顶] Hibernate从入门到精通(七)多对一单向关联映射

    上次的博文Hibernate从入门到精通(六)一对一双向关联映射中我们介绍了一下一对一双向关联映射,本次博文我们讲解一下多对一关联映射 多对一单向关联映射 多对一关联映射与一对一关联映射类似,只是在多 ...

  7. [置顶] Hibernate从入门到精通(十)多对多单向关联映射

    上一篇文章Hibernate从入门到精通(九)一对多双向关联映射中我们讲解了一下关于一对多关联映射的相关内容,这次我们继续多对多单向关联映射. 多对多单向关联映射 在讲解多对多单向关联映射之前,首先看 ...

  8. Hibernate 单项多对一的关联映射

    在日常开发中会出现很对多对一的情况,本文介绍hibernate中多对一的关联映射. 1.设计表结构 2.创建student对象 3.创建Grade对象 4.写hbm.xml文件 5.生成数据库表 生成 ...

  9. Hibernate之关于多对多单向关联映射

    [Hibernate]之关于多对多单向关联映射 老师和学生,最典型的多对多关联, Teacher和Student.所谓单向意思就是说.老师知道自己的教的是哪些学生而学生不知道是哪些老师教. 也能够这么 ...

随机推荐

  1. 原生JavaScript实现页面回到顶部的功能

    /*如果想实现点击一个按钮让滚动条回到最顶部的功能,首先可能就会想到它是从底部位置移动到顶部的位置 它是一个运动的过程,只要知道当前位置(current Position)和想要到达的位置(targe ...

  2. api-gateway实践(08)新服务网关 - 云端发布和日志查看

    一.发布应用 1.新建应用空间 1.1.新建应用空间 1.2.新建应用 1.3.上传程序包 2.创建应用引擎服务 3.发布应用 3.1.为应用容器绑定Web运行环境(应用引擎服务) 3.2.发布应用( ...

  3. Python学习之中文注释问题

    简单写个输入.输出,并注释 # 输入 print'100+200=',100+200 # 输入 name = raw_input() 报错了: SyntaxError: Non-ASCII chara ...

  4. 清除session信息

    session.removeAttribute("sessionname")是清除SESSION里的某个属性.     session.invalidate()是让SESSION失 ...

  5. 测试驱动开发实践5————testSave之修改文档分类

    [内容指引] 1.确定"修改文档分类"的微服务接口及所需的参数 2.设计测试用例及测试用例合并 3.为测试用例赋值并驱动开发 上一篇我们通过17个测试用例完成了"新增文档 ...

  6. tornado框架源码分析---Application类之debug参数

    先贴上Application这个类的源码. class Application(httputil.HTTPServerConnectionDelegate): """A ...

  7. 招募:Wiki 文档翻译小伙伴招募

    https://github.com/dotnetcore/CAP/issues/93 为了推进 CAP 的国际化工作,为全球其他 .NET 开发者提供更加良好的文档阅读体验,现在需要对CAP wik ...

  8. C++程序设计语言(特别版) -- 一个桌面计算器

    前言 这里要介绍各种语句和表达式,将通过一个桌面计算器的程序做些事情,该计算器提供四种座位浮点数的中缀运算符的标准算术运算. 这个计算器由四个部分组成:一个分析器,一个输入函数,一个符号表和一个驱动程 ...

  9. Python系列之 - python运算符

    废话不多说,上节说的是数据类型,本篇讲讲数据运算. 在算式"1+2"中,"1"和"2"被称为操作数,"+"被称为运算符 ...

  10. vue 2.0 路由切换以及组件缓存源代码重点难点分析

    摘要 关于vue 2.0源代码分析,已经有不少文档分析功能代码段比如watcher,history,vnode等,但没有一个是分析重点难点的,没有一个是分析大命题的,比如执行router.push之后 ...