Hibernate单向多对一映射关系01和级联保存和Junit4使用

一、了解什么是一对多?

     一对多这种关系用的很多,但是很多人对此模糊不清,一对多不也就是多对一吗,多对多中特殊的一个不也就是一对多吗,为什么还分的那么清楚?并且还是什么单向和双向的那么麻烦,其实原因很简单,就是针对不同的情况和业务需求而产生的这种说法

     例如:学生和班级

        单向一对多:一个班级对应多个学生。 也就是说,在某种业务情况下,我只需要知道从一个班级中知道有哪些学生,但是我不需要知道一个学生在哪个班级,这时候我就没必要写通过学生能查找到对应班级这个业务

(站在“”这一边,可以通过班级看这个班级的学生功能,没有通过学生知道班级功能)

        单向多对一:多个学生对应一个班级,可以通过每个学生查找到所在的班级名称,而不能从班级中查找到有哪些学生在里面,这就是从多到一的单面考虑,也就是说,当我们的业务需求只需要通过学生能找到对应班级,而不需要通过班级知道有哪些学生的时候,就可以写这样的单向多对一的关系映射

(站在“多”这一边,有通过学生知道班级功能,没有通过班级看这个班级的学生功能)

        双向一对多/双向多对一:这两个是一个意思,既然度双向了,说明不管从哪一方去找另一方,度可以找得到,也就无所谓一对多,还是多对一了。从这个学生和班级来讲,通过学生能知道他所在的班级,通过班级,能知道该班级下所有的学生。

      通过这个例子就知道了单向和双向是干嘛用的,就是根据不同的业务所规定的,如果你需要双向就写双向的映射关系,如果只需要从一方到另一方,那么就写自己所需要的,单向一对多或者单向多对一。

     

    所以我们接下要讲解的就是这三种,单向一对多、单向多对一、双向一对多。

二、单向多对一

      多个用户属于同一个组,多个学生属于同一个班级,多个学生属于同一个宿舍,等等,很多关系是这种多对一。

      多个学生属于同一个班级。单向多对一    

                  UserClasses类的关系图。

               d90b4caac0659a6e292da5dcf2aa7e15e5a.jpg

知道需要的功能是多对一之后,站在“多的角度”,在设计完学生类基本属性的情况下,增加一个“班级类”来保存班级信息,就是说该学生数据有这个班级表信息

例子

9d897e35c06fd3d1f773427895a2468a8e5.jpg

1、首先创建班级和学生实体类

Classes.java

package com.java.model;

public class Classes {
	private String name;
	private long id;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public long getId() {
		return id;
	}
	public void setId(long id) {
		this.id = id;
	}
	
}

配置Classes.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>
       <!--班级表class name对应类名,table对应数据库中的表名  -->
	<class name="com.java.model.Classes" table="t_classes">
	
	<!--主键id name对应实体类的属性,column对应数据库表中的字段 ,generator主键 -->
	<id name="id" column="class_id">
		<generator class="native"></generator>
	</id>
	
		<!-- 该对象的基本属性 -->
	<property name="name" column="class_name">
	</property>
	</class>


</hibernate-mapping>

 

因为是多对一,站在“多”的角度,也就站在学生角度,classes 班级 不需要有那种关系,只需要配置自己的属性就行了

 

Student.java

package com.java.model;

public class Student {
	private String name;
	private long id;
	
	//定义一个班级对象
	private Classes c;

	public Classes getC() {
		return c;
	}
	public void setC(Classes c) {
		this.c = c;
	}
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public long getId() {
		return id;
	}
	public void setId(long id) {
		this.id = id;
	}
	

	
	@Override
	public String toString() {
		return "Student [name=" + name + ", id=" + id + ", c=" + c + "]";
	}
	
	

}

 Student需要有相应的能力,所以需要配置外键关系。many to one

Student.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>
       <!--学生表class name对应类名,table对应数据库中的表名  -->
	<class name="com.java.model.Student" table="t_student">
	
	<!--主键id name对应实体类的属性,column对应数据库表中的字段 ,generator主键 -->
	<id name="id" column="student_id">
		<generator class="native"></generator>
	</id>
	
	<!-- 该对象的基本属性 -->
	<property name="name" column="student_name">
	</property>
	
	
	 <!-- 重点在这里 
	 		学生表外键和班级表主键相关联,
            name:学生实体类中定义的那个对象的引用名称这东西:private Class c;。
            column:学生表中的外键名称。注意,是被外键约束的字段的名称,写这些配置文件,要时刻记得那两张数据库表的关系。
			学生表外键名可以取和班级表主键名不一样的名字
			本例学生表外键名:classId  
			本例班级表主键名:class_id  
         -->
	<many-to-one name="c" column="classId" class="com.java.model.Classes"></many-to-one>
	</class>


</hibernate-mapping>

单向多对一的关系映射就是这样,原理就是在数据库中多方设置外键指向一方。 然后通过映射文件来利用这个外键帮我们达到我们的目的,这里就是通过外键,找到对应的班级放入我们自己类中的classes属性中。

many-to-one的属性

        name:影射类属性的名称

        class:非必须的,不写,hibernate会自己根据name的值去查找

        column:关联的字段,也是非必须的,hibernate也会帮我们在查找,但是不写的话,User中的外键名称必须和Classes中的主键名称相同。也就是关联的字段名称要相同,这样才找得到

        net-null:默认是false,就是不能为空

        5c7bf8a3410d4c80e3b9485baa8b3b181c3.jpg

 

然后执行/StudentTest.java

package com.java.service;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import com.java.util.HibernateUtil;
public class StudentTest {

	public static void main(String[] args) {
		SessionFactory sessionFactory=HibernateUtil.getSessionFactory();
		  Session session=sessionFactory.openSession(); // 生成一个session
		    session.beginTransaction(); // 开启事务
		    
		  
		    
		    session.getTransaction().commit(); // 提交事务
		    session.close(); // 关闭session
	}
}

查看数据库,发现自动生成了表,还有主外键的关联关系,神奇

ebed66c5c9fab43b9d2d63420933b760f75.jpg

三、“多对一”级联保存

8276647ed5435d2dd49eae04a54896dbe1c.jpg

一般我们写入数据进数据步骤是:

1.操作班级表,给班级表创建一条班级数据

2.操作学生表,给学生表创建一条学生数据,然后放入班级信息

package com.java.service;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import com.java.model.Classes;
import com.java.model.Student;
import com.java.util.HibernateUtil;
public class StudentTest {

	public static void main(String[] args) {
		SessionFactory sessionFactory=HibernateUtil.getSessionFactory();
		  Session session=sessionFactory.openSession(); // 生成一个session
		    session.beginTransaction(); // 开启事务

		    //写好班级信息
		    Classes banji=new Classes();
		    banji.setName("15软件工程");
                    //持久化,操作班级表,把班级信息放到数据库
		    session.save(banji);
		     
                    //写好学生表的信息
		    Student s1=new Student();
		    s1.setName("张三");
		    s1.setC(banji);
                    //持久化,操作学生表,把学生信息放到数据库
		     session.save(s1);


		    Student s2=new Student();
		    s2.setName("李四");
                    s2.setC(banji);
		    session.save(s2);
		    
		    session.getTransaction().commit(); // 提交事务
		    session.close(); // 关闭session
	}
}

因为我们设置了单向多对一,也就是说,我们可以只操作学生表,同时操作班级表

当我们在学生表保存学生信息和班级信息,班级表自动保存,级联保存先配置“多”的这边

3a94f7b86f38b4eed7dadba1a90d1b59d32.jpg

<?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>
       <!--学生表class name对应类名,table对应数据库中的表名  -->
	<class name="com.java.model.Student" table="t_student">
	
	<!--主键id name对应实体类的属性,column对应数据库表中的字段 ,generator主键 -->
	<id name="id" column="student_id">
		<generator class="native"></generator>
	</id>
	
	<!-- 该对象的基本属性 -->
	<property name="name" column="student_name">
	</property>
	
	
	 <!-- 重点在这里 
	 		学生表外键和班级表主键相关联,
            name:学生实体类中定义的那个对象的引用名称:private Class c;。
            column:学生表中的外键名称。注意,是被外键约束的字段的名称,写这些配置文件,要时刻记得那两张数据库表的关系。
			学生表外键名可以取和班级表主键名不一样的名字
			本例学生表外键名:classId  
			本例班级表主键名:class_id  
         -->
	<many-to-one name="c" column="classId" class="com.java.model.Classes" cascade="save-update"></many-to-one>
	</class>


</hibernate-mapping>

现在就可以级联保存,保存学生表的数据,班级表数据就自动生成

package com.java.service;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import com.java.model.Classes;
import com.java.model.Student;
import com.java.util.HibernateUtil;
public class StudentTest {

	public static void main(String[] args) {
		SessionFactory sessionFactory=HibernateUtil.getSessionFactory();
		  Session session=sessionFactory.openSession(); // 生成一个session
		    session.beginTransaction(); // 开启事务

		    //写好班级信息对象
		    Classes banji=new Classes();
		    banji.setName("15软件工程");
                    		     
                    //写好学生的信息对象		    
                    Student s1=new Student();
		    s1.setName("张三");
                    //把班级信息对象加到学生的信息对象
		    s1.setC(banji);

                    //持久化,只操作学生表,把学生信息对象放到学生表
		     session.save(s1);


		    Student s2=new Student();
		    s2.setName("李四");
                    s2.setC(banji);
		    session.save(s2);
		    
		    session.getTransaction().commit(); // 提交事务
		    session.close(); // 关闭session
	}
}

这样添加学生信息时把班级信息写上点击确定就行,自动把班级信息保存。

不用先保存班级信息,才能保存学生信息。直接保存学生信息即可

数据库:

学生表

473d7db4d2bf44629c7928cf03f2e9cc1be.jpg

班级表

498860473bba9c9aa58b168ca525226f2ad.jpg

 

四、Junit4使用

Junit方法如下

1a750f2a6b5dec7f542fda0e344fdc6cd46.jpg

使用这些方法看先后执行顺序

8d01972ff1c5863500fd435b5e08712303f.jpg

670637966be86bd04fc60538988d5333580.jpg

StudentTest2 

package com.java.service;

import static org.junit.Assert.*;

import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test;

public class StudentTest2 {

	@BeforeClass
	public static void setUpBeforeClass() throws Exception {
		System.out.println("类初始化前调用");
	}

	@AfterClass
	public static void tearDownAfterClass() throws Exception {
		System.out.println("类初始化前调用");
	}

	@Before
	public void setUp() throws Exception {
		System.out.println("测试方法前调用");
	}

	@After
	public void tearDown() throws Exception {
		System.out.println("测试方法后调用");
	}

	@Test
	public void test() {
		System.out.println("测试方法");
	}

}

执行顺序结果

9a75ce06840c0e44170e55cb2c08d78e006.jpg

常用的就是测试方法前后调用

72797366d904806780234748a49918da2d4.jpg

package com.java.service;

import static org.junit.Assert.*;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

import com.java.model.Classes;
import com.java.model.Student;
import com.java.util.HibernateUtil;

public class StudentTest2 {

	private SessionFactory sessionFactory=HibernateUtil.getSessionFactory();
	private Session session;
	
	@Before
	public void setUp() throws Exception {
	    session=sessionFactory.openSession(); // 生成一个session
	    session.beginTransaction(); // 开启事务
	}

	@After
	public void tearDown() throws Exception {
		 session.getTransaction().commit(); // 提交事务
		 session.close(); // 关闭session
	}
	
	@Test
	public void testSaveClassAndStudent() {
		System.out.println("正常思路保存数据");

	    //写好班级信息
	    Classes banji=new Classes();
	    banji.setName("15软件工程");
                //持久化,操作班级表,把班级信息放到数据库
	    session.save(banji);
	     
                //写好学生表的信息
	    Student s1=new Student();
	    s1.setName("张三");
	    s1.setC(banji);
                //持久化,操作学生表,把学生信息放到数据库
	     session.save(s1);


	    Student s2=new Student();
	    s2.setName("李四");
                s2.setC(banji);
	    session.save(s2);

	}
	@Test
	public void testSaveClassAndStudentWithCascade() {
		System.out.println("级联保存");

		 //写好班级信息
	    Classes banji=new Classes();
	    banji.setName("15软件工程");
                		     
                //写好学生的信息		    
                Student s1=new Student();
	    s1.setName("张三");
	    s1.setC(banji);

                //持久化,只操作学生表,把学生信息放到学生表
	     session.save(s1);


	    Student s2=new Student();
	    s2.setName("李四");
                s2.setC(banji);
	    session.save(s2);


	}

}

想测试哪个方法就进行如下操作:

9b412910ed96f228ff26e183457346166a2.jpg

 

五、单向多对一 

          说了单向多对一,现在来说说单向一对多,根据前面我们讲的,其实就是换了一个角度去想这个问题,学生和班级,现在站在班级这方去看,肯定是需要通过班级能知晓所有在这个班学习的同学的信息。  

         注意方向:站在“一”的角度看待问题,班级是“一”,就是说每一个班级里都有几十个学生信息,通过班级这个表我可以找到某个学生信息

站在“一”的角度,在设计完班级类基本属性的情况下,增加一个“学生类的集合”(每个学生类代表一个学生)

在model的加学生集合属性

private Set<Student> students=new HashSet<Student>();

 

    788a4336bbb14d25ca2b03702b84f1c29f5.jpg   

       数据库中还是没变,不管是单向一对多还是单向多对一,永远都市通过外键来维护关系的,单向多对一:通过映射文件知道,User的外键能够查找到对应的班级。单向一对多:又是怎么来实现通过User的外键找到属于本班的所有学生的呢?

看代码

Classes.java

package com.java.model;

import java.util.HashSet;
import java.util.Set;

public class Classes {
	private String name;
	private long id;
        <!--重点-->
	private Set<Student> students=new HashSet<Student>();
	
	
	public Set<Student> getStudents() {
		return students;
	}
	public void setStudents(Set<Student> students) {
		this.students = students;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public long getId() {
		return id;
	}
	public void setId(long id) {
		this.id = id;
	}
	
}

  Classes.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>
       <!--班级表class name对应类名,table对应数据库中的表名  -->
	<class name="com.java.model.Classes" table="t_classes">
	
	<!--主键id name对应实体类的属性,column对应数据库表中的字段 ,generator主键 -->
	<id name="id" column="class_id">
		<generator class="native"></generator>
	</id>
	
		<!-- 该对象的基本属性 -->
	<property name="name" column="class_name">
	</property>
	
	
	
        <!--关键在这里,
            name:set集合属性的名称(model里的classes类的学生集合名称)
            key/column:这个是外键名,这个外键字段名不一定要和本类的主键字段名相同,这点要搞清楚,记住数据库表的关系,谁指向谁就不会混淆
                        这个外键值放在学生表里(不管是单向一对多还是单向多对一,都需要在多方加上外键)
            one to many/class:一对多,所映射的类名(全限定类名,直接写类名也可以,hibernate会帮我们自动写好)
        -->
		        <set name="students">
		            <key column="cid"></key>
		            <one-to-many class="com.java.model.Student" />
		        </set>
		</class>
</hibernate-mapping>

         通过映射文件可以知道,通过本类(班级)中的主键值去寻找映射类(学生)中的外键值,有匹配的就将其对象放置到set集合中来。所以说,不管是单向一对多还是单向多对一,都需要在多方加上外键,也就是说,他们的原理度是一样的,只是站的角度不一样,单向一对多站的角度在于一方,一方如何通过外键来达到自己的目的,就看自己的映射文件如何编写,单向多对一站的角度在多方,多方如何通过外键来达到自己的目的,也要看自己的映射文件的编写。  

    由于是一对多,站在“一”的角度,学生端正常设置就行

Student.java

package com.java.model;

public class Student {
	private String name;
	private long id;
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public long getId() {
		return id;
	}
	public void setId(long id) {
		this.id = id;
	}
	
}

Student.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>
       <!--学生表class name对应类名,table对应数据库中的表名  -->
	<class name="com.java.model.Student" table="t_student">
	
	<!--主键id name对应实体类的属性,column对应数据库表中的字段 ,generator主键 -->
	<id name="id" column="student_id">
		<generator class="native"></generator>
	</id>
	
	<!-- 该对象的基本属性 -->
	<property name="name" column="student_name">
	</property>
	

	</class>


</hibernate-mapping>

 

测试方法(把数据库表删了,因为前面生成了“多对一”的表,现在测试“一对多”)

StudentTest.java

@Test
public static void main(String[] args) {
		SessionFactory sessionFactory=HibernateUtil.getSessionFactory();
		  Session session=sessionFactory.openSession(); // 生成一个session
		    session.beginTransaction(); // 开启事务
		    
		   
		    
		    session.getTransaction().commit(); // 提交事务
		    session.close(); // 关闭session
	}

发现数据库生成成功

0bce5acab395ed16495fc95d9818d5499f5.jpg

、“一对多”级联保存

保存班级表的时候带上学生信息,只操作班级表,学生表也会保存。

首先给Classes.hbm.xml配置cascade="save-update"

3f883d09df9ac1e91c916f62d67b6738df7.jpg

然后测试方法

       @Test
	public void testSaveClassAndStudent() {
		//写好班级信息对象
		Classes c=new Classes();
	    c.setName("08计本");
	    //写好学生信息对象
	    Student s1=new Student();
	    s1.setName("张三");
	    
	    Student s2=new Student();
	    s2.setName("李四");
	    
	    //把学生对象加到班级对象
	    c.getStudents().add(s1);
	    c.getStudents().add(s2);
	    
	    //将班级对象保存到班级表中,学生表自动更新
	    session.save(c);
	}

结果

0b0f41186720f75b7dfcd8a31b3ef5db81b.jpg1255b558034d61f43ea131efcdd0024c6ea.jpg

 

 

转载于:https://my.oschina.net/u/3848699/blog/2222912

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值