第四节:班级学生一对多映射实现(双向)                                
查询班级的时候能够获取所有的学生;
 
在上一节的基础之上;我们在Class端也保存学生的关系;
com.cy.model.Class:
public class Class {
private long id;
private String name;
private Set<Student> students = new HashSet<Student>(); public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Set<Student> getStudents() {
return students;
}
public void setStudents(Set<Student> students) {
this.students = students;
}
}

Class.hbm.xml:

<hibernate-mapping package="com.cy.model">
<class name="Class" table="t_class">
<id name="id" column="classId">
<generator class="identity"></generator>
</id>
<property name="name" column="className"></property>
<set name="students" cascade="save-update">
<!-- key外键,column是对应Student表的外键classId
就是many-to-one中的column;
-->
<key column="classId"></key>
<!-- class就是students属性,对应的集合中装的类Student -->
<one-to-many class="com.cy.model.Student"/>
</set>
</class>
</hibernate-mapping>

com.cy.model.Student还和之前一样:

package com.cy.model;

public class Student {
private long id;
private String name;
private Class c; public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
} public Class getC() {
return c;
}
public void setC(Class c) {
this.c = c;
} @Override
public String toString() {
return "Student [id=" + id + ", name=" + name + "]";
} }

Student.hbm.xml还和之前一样:

<hibernate-mapping package="com.cy.model">
<class name="Student" table="t_student">
<id name="id" column="stuId">
<generator class="identity"></generator>
</id>
<property name="name" column="stuName"></property>
<many-to-one name="c" column="classId" class="com.cy.model.Class" cascade="save-update"></many-to-one>
</class>
</hibernate-mapping>

测试代码StudentTest:

@Test
public void testSaveClassAndStudent() {
Class c = new Class();
c.setName("08计本"); Student s1 = new Student();
s1.setName("张三");
Student s2 = new Student();
s2.setName("李四"); c.getStudents().add(s1);
c.getStudents().add(s2); //这里因为保存Class时,class引用了临时状态的s1、s2;会保存失败
//必须在Class.hbm.xml中配置<set name="students" cascade="save-update"级联保存更新才可以。
session.save(c);
} /**
* 通过班级端,查找学生
*/
@Test
public void getStudentsByClass() {
Class c = (Class) session.get(Class.class, Long.valueOf(2));
Set<Student> students = c.getStudents(); Iterator<Student> it = students.iterator();
while(it.hasNext()){
Student s = it.next();
System.out.println(s);
} //Student [id=1, name=李四]
//Student [id=2, name=张三]
}

保存成功!查询成功!

发出的sql:

第五节:inverse 属性                                      
inverse属性主要是在一对多、多对一双向关系中,由一端来维护主外键关系;
 
测试代码:
@Test
public void testAdd(){
Class c=new Class();
c.setName("09计本"); Student s1=new Student();
s1.setName("王五"); session.save(c);
session.save(s1);
} @Test
public void testInverse(){
/**
* Class和Student都是从session中获取的,都已经是持久化对象;
* 对着两个持久化对象进行学生设置班级、班级也设置学生
* 持久化对象设置关系,hibernate session检测到之后就会进行持久化操作(同步数据库)
*/
Class c=(Class)session.get(Class.class, Long.valueOf(1));
Student s=(Student)session.get(Student.class, Long.valueOf(1)); s.setC(c);
c.getStudents().add(s); /**
* 可以看到学生设置班级、班级添加学生,两端都维护了这个关系,都维护了这个外键。
引入inverse属性,只在某一端维护这个主外键关系。通常是在多的一端(one-to-many)设置inverse属性。
这里就是在学生Student端添加inverse=true属性
*/
}

1.先执行testAdd,再执行testInverse:

由于对student、class两个持久化的对象进行,互设关系的操作,hibernate session检测到之后就会进行持久化操作(同步数据库):

确实数据库发现已经设置了外键关系,打印:

两端都来维护这个关系,发出两条sql语句,有点冗余,引入inverse:

Class.hbm.xml:

<hibernate-mapping package="com.cy.model">
<class name="Class" table="t_class">
<id name="id" column="classId">
<generator class="identity"></generator>
</id>
<property name="name" column="className"></property>
<set name="students" cascade="save-update" inverse="true">
<!-- key外键,column是对应Student表的外键classId
就是many-to-one中的column;
-->
<key column="classId"></key>
<!-- class就是students属性,对应的集合中装的类Student -->
<one-to-many class="com.cy.model.Student"/>
</set>
</class>
</hibernate-mapping>

2.再次先执行testAdd,再执行testInverse:

只有一条sql了,只在多的一方来维护这个外键关系;

第六节:级联删除                                              
删除班级的时候,我们把学生也删掉;
/**
* 级联删除
* 删除班级的时候,级联删除学生
*/
@Test
public void testDeleteClassCascade(){
Class c=(Class)session.get(Class.class, Long.valueOf(1));
session.delete(c);
}
默认是不能级联删除的,因为删除class的时候,classId被student表外键关联;
会报错:

设置cascade=delete就可以级联删除了;
修改Class.hbm.xml配置如下:
<hibernate-mapping package="com.cy.model">
<class name="Class" table="t_class">
<id name="id" column="classId">
<generator class="identity"></generator>
</id>
<property name="name" column="className"></property>
<set name="students" cascade="delete" inverse="true">
<key column="classId"></key>
<one-to-many class="com.cy.model.Student"/>
</set>
</class>
</hibernate-mapping>

console:

注:级联删除是一个危险的操作,企业开发一般是不允许的;很可能导致其他业务数据的丢失。。。

第七节:一对多双向自身关联关系映射                                

例如使用在菜单上,使用节点来模拟:

com.cy.model.Node:

package com.cy.model;

import java.util.HashSet;
import java.util.Set; public class Node {
private long id;
private String name; private Node parentNode; //它的父节点
private Set<Node> childNodes = new HashSet<Node>(); //它的子节点 public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Node getParentNode() {
return parentNode;
}
public void setParentNode(Node parentNode) {
this.parentNode = parentNode;
}
public Set<Node> getChildNodes() {
return childNodes;
}
public void setChildNodes(Set<Node> childNodes) {
this.childNodes = childNodes;
} }

Node.hbm.xml:

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.cy.model">
<class name="Node" table="t_node">
<id name="id" column="nodeId">
<generator class="native"></generator>
</id>
<property name="name" column="nodeName"></property> <many-to-one name="parentNode" column="parentId" class="com.cy.model.Node" cascade="save-update"></many-to-one> <set name="childNodes" inverse="true">
<key column="parentId"></key>
<one-to-many class="com.cy.model.Node"/>
</set>
</class>
</hibernate-mapping>

将Node.hbm.xml加入hibernate.cfg.xml中mapping配置;

测试代码:

package com.cy.service;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.junit.After;
import org.junit.Before;
import org.junit.Test; import com.cy.model.Node;
import com.cy.util.HibernateUtil; public class NodeTest {
private SessionFactory sessionFactory = HibernateUtil.getSessionFactory();
private Session session; @Before
public void setUp() throws Exception {
session = sessionFactory.openSession();
session.beginTransaction();
} @After
public void tearDown() throws Exception {
session.getTransaction().commit();
session.close();
} @Test
public void testSaveMenu() {
Node node=new Node();
node.setName("根节点"); Node subNode1=new Node();
subNode1.setName("子节点1"); Node subNode2=new Node();
subNode2.setName("子节点2"); subNode1.setParentNode(node);
subNode2.setParentNode(node); session.save(subNode1);
session.save(subNode2);
}
}

结果:

Hibernate学习4—关联关系一对多映射2的更多相关文章

  1. Hibernate学习4—关联关系一对多映射

    一.班级学生一对多映射实现(单向) Class 1 * Student 一对多 在学生多的一方,配置一的对象:   com.cy.model.Class: package com.cy.model; ...

  2. 码农小汪-Hibernate学习8-hibernate关联关系注解表示@OneToMany mappedBy @ManyToMany @JoinTable

    近期我也是有点郁闷,究竟是程序中处理关联关系.还是直接使用外键处理关联关系呢?这个的说法不一致!程序中处理这样的关联关系的话.自己去维护这样的约束.这样的非常乐观的一种做法!或者是直接在数据库中处理这 ...

  3. Hibernate学习之双向一对多映射(双向多对一映射)

    © 版权声明:本文为博主原创文章,转载请注明出处 1.双向映射与单向映射 - 一对多单向映射:由一方(教室)维护映射关系,可以通过教室查询该教室下的学生信息,但是不能通过学生查询该学生所在教室信息: ...

  4. Hibernate学习之单向一对多映射

    © 版权声明:本文为博主原创文章,转载请注明出处 说明:该实例是通过映射文件和注解两种方式实现的.可根据自己的需要选择合适的方式 实例: 1.项目结构 2.pom.xml <project xm ...

  5. Hibernate学习笔记(二)

    2016/4/22 23:19:44 Hibernate学习笔记(二) 1.1 Hibernate的持久化类状态 1.1.1 Hibernate的持久化类状态 持久化:就是一个实体类与数据库表建立了映 ...

  6. Hibernate学习(二)关系映射----基于外键的单向一对一

    事实上,单向1-1与N-1的实质是相同的,1-1是N-1的特例,单向1-1与N-1的映射配置也非常相似.只需要将原来的many-to-one元素增加unique="true"属性, ...

  7. Hibernate学习之——搭建log4j日志环境

    昨天讲了Hibernate开发环境的搭建以及实现一个Hibernate的基础示例,但是你会发现运行输出只有sql语句,很多输出信息都看不见.这是因为用到的是slf4j-nop-1.6.1.jar的实现 ...

  8. Hibernate学习笔记(一)

    2016/4/18 19:58:58 Hibernate学习笔记(一) 1.Hibernate框架的概述: 就是一个持久层的ORM框架. ORM:对象关系映射.将Java中实体对象与关系型数据库中表建 ...

  9. Hibernate 学习笔记一

    Hibernate 学习笔记一 今天学习了hibernate的一点入门知识,主要是配置domain对象和表的关系映射,hibernate的一些常用的配置,以及对应的一个向数据库插入数据的小例子.期间碰 ...

随机推荐

  1. poj1273&&hdu1532

    题解: 双倍经验 dinic模板题 然后ff过不了... 代码: #include<cstdio> #include<cstring> #include<algorith ...

  2. jQuery实现点击式选项卡

    参考:jQuery权威指南jQuery初步jQuery选择器jQuery操作domjQuery操作dom事件jQuery插件jQuery操作AjaxjQuery动画与特效jQuery实现导航栏jQue ...

  3. C++面向对象高级编程(二)基础篇

    技术在于交流.沟通,转载请注明出处并保持作品的完整性. 概要 知识点1.重载成员函数 知识点2 . return by value, return by reference 知识点3 重载非成员函数 ...

  4. Linux文件操作及管理

    ---恢复内容开始--- 一.Linux系统的结构 1.Linux是一个倒树型结构,最大的目录名称为“/”(根目录) 2.Linux系统的二级目录 /bin     ##binary二进制可执行文件, ...

  5. Join, Group Join

    Linq的 Join对应SQL中的inner join,当左右两张表有匹配数据时才返回一条记录: Linq的 Group Join对应SQL中的LEFT OUTER JOIN,即使右表中没有匹配,也从 ...

  6. jenkins执行xctool命令出现command not found问题解决方法

    1.控制台执行 echo $PATH 把输出的这句话复制 2.jenkins->系统管理->系统设置 勾选Environment variables,添加键值,键:PATH,值:刚才复制的 ...

  7. JDK1.7之 HashMap 源码分析

    转载请注明出处:http://blog.csdn.net/crazy1235/article/details/75451812 类继承关系 构造函数 Entry put put putForNullK ...

  8. 查看PHP以字母"E"开头的常量

    1.E_ALL <?php echo E_ALL; ?> 32767 2.E_COMPILE_ERROR <?php echo E_COMPILE_ERROR; ?> 64 3 ...

  9. python 多维list声明时的小问题

    a=[[]]*3 a Out[18]: [[], [], []] a[0].append(1) a Out[20]: [[1], [1], [1]] b=[[] for _ in range(3)] ...

  10. sql server不能删除数据库,显示错误:正在使用

    解决办法: use mastergoalter database database_name set single_user with rollback immediate --将数据库回滚到原始配置 ...