Hibernate关系映射---单向一对多curd(增删查改)操作实例

Hibernate关系映射---单向一对多curd(增删查改)操作实例

实际情况:一个部门下面拥有多个员工,所以部门和员工之间是一对多的关系

1、新建一个Java工程,添加hibernate3.3支持,连接test数据库

2、在com.etc.dao下新建一个类Dept.java如下:

package com.etc.dao;

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

public class Dept {
	private int no;
	private String name;
	//在一端加入多端的set集合
	Set<Emp> emps = new HashSet<Emp>();
	public int getNo() {
		return no;
	}
	public void setNo(int no) {
		this.no = no;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Set<Emp> getEmps() {
		return emps;
	}
	public void setEmps(Set<Emp> emps) {
		this.emps = emps;
	}
	@Override
	public String toString() {
		return "Dept [no=" + no + ", name=" + name + ", emps=" + emps + "]";
	}
}

3、在com.etc.dao下新建一个类Emp.java如下:

package com.etc.dao;

public class Emp {
	private int no;
	private String name;
	public int getNo() {
		return no;
	}
	public void setNo(int no) {
		this.no = no;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	@Override
	public String toString() {
		return "Emp [no=" + no + ", name=" + name + "]";
	}
}

4、在com.etc.dao下新建Dept类的映射文件Dept.hbm.xml如下:

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
	<class	name="com.etc.dao.Dept" table="t_dept" catalog="test">
		<id name="no">
			<column name="dept_no"/>
			<generator class="native"/>
		</id>
		<property name="name">
			<column name="dept_name"/>
		</property>
		<set name="emps" cascade="all" fetch="join">
			<key>
				<column name="dept_no" />
			</key>
			<one-to-many class="com.etc.dao.Emp"/>
		</set>
	</class>
</hibernate-mapping>

5、在com.etc.dao下新建Emp类的映射文件Emp.hbm.xml如下:

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
	<class name="com.etc.dao.Emp" table="t_emp" catalog="test">
		<id name="no">
			<column name="emp_no"/>
			<generator class="native"/>
		</id>
		<property name="name">
			<column name="emp_name"/>
		</property>
	</class>
</hibernate-mapping>

6、配置hibernate.cfg.xml文件:

<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
          "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<!-- Generated by MyEclipse Hibernate Tools.                   -->
<hibernate-configuration>
	<session-factory>
		<property name="show_sql">true</property>
		<property name="format_sql">true</property>
		<!-- java类中使用getCurrentSession()方法时需要配置current_session_context_class -->
		<property name="current_session_context_class">thread</property>
		<!-- 表不存在时,会自动创建表,刚开始的时候执行一次便可 -->
		<!-- <property name="hbm2ddl.auto">create</property> -->
		<property name="dialect">
			org.hibernate.dialect.MySQLDialect
		</property>
		<property name="connection.url">
			jdbc:mysql://localhost:3306/test
		</property>
		<property name="connection.username">root</property>
		<property name="connection.password">root</property>
		<property name="connection.driver_class">
			com.mysql.jdbc.Driver
		</property>
		<property name="myeclipse.connection.profile">
			com.mysql.jdbc.Driver
		</property>
		<!-- 配置映射文件 -->
		<mapping resource="com/etc/dao/Dept.hbm.xml"/>
		<mapping resource="com/etc/dao/Emp.hbm.xml"/>
	</session-factory>

</hibernate-configuration>

7、新建test文件夹,新建com.etc.dao包,在包下新建DeptDAOTest.java测试类如下:

package com.etc.dao;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import org.junit.Test;

public class DeptDAOTest {
	/**
	 * Description:新增:新增部门,并且新增部门下的员工,由一端操作多端(将多端加入一端中,使用session保存一端信息,
	 * 并且级联保存多端信息),设置cascade=true
	 * @author zoey
	 * @date 2017年7月27日
	 */
	@Test
	public void testCreate(){
		SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();
		Session session = sessionFactory.getCurrentSession();
		session.beginTransaction();
		Dept dept = new Dept();
		dept.setName("技术部");
		
		Emp emp = new Emp();
		emp.setName("张三");
		Emp emp2 = new Emp();
		emp2.setName("李四");
		
		dept.getEmps().add(emp);
		dept.getEmps().add(emp2);
		
		session.save(dept);
		session.getTransaction().commit();
	}
	/**
	 * Description:修改:修改部门,并且修改部门下的员工姓名(直接调用set方法就可以,不需要调用session的
	 * 更新方法)
	 * @author zoey
	 * @date 2017年7月27日
	 */
	@Test
	public void testUpdate(){
		SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();
		Session session = sessionFactory.getCurrentSession();
		session.beginTransaction();
		Dept dept = (Dept)session.get(Dept.class,1);
		dept.setName("开发部");
		
		for(Emp emp:dept.getEmps()){
			emp.setName("小明");
		}
		session.getTransaction().commit();
	}
	/**
	 * Description:查询:查询部门,并且查询该部门下的所有员工,一般是发起两条sql,如果想要提高效率,
	 * 发起一条sql 可以在Dept.hbm.xml中set中配置fetch=join,就会关联查询,提高效率
	 * @author zoey
	 * @date 2017年7月27日
	 */
	@Test
	public void testRetrieve(){
		SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();
		Session session = sessionFactory.getCurrentSession();
		session.beginTransaction();
		Dept dept = (Dept)session.get(Dept.class, 1);
		System.out.println(dept);
		//得到部门中的所有的员工
		for(Emp emp:dept.getEmps()){
			System.out.println(emp);
		}
		session.getTransaction().commit();
	}
	/**
	 * Description:删除:删除部门的时候,同时删除该部门下的所有员工(设置了cascade=all)
	 * @author zoey
	 * @date 2017年7月27日
	 */
	@Test
	public void testDelete(){
		SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();
		Session session = sessionFactory.getCurrentSession();
		session.beginTransaction();
		
		Dept dept = (Dept)session.get(Dept.class, 1);
		session.delete(dept);
		
		session.getTransaction().commit();
	}
}


总结:

1、增加(Create):调用session.save()方法保存部门的信息时,同时保存该部门下的员工信息,需要在Dept.hbm.xml

映射文件中的set属性中设置cascade=all,才能实现级联保存。如果不设置cascade=all属性,那么只会保存该部门信息,不会保存该部门下的员工信息,需要再次调用session.save()方法来保存员工信息,所以,一般我们建议使用cascade=all属性,使得保存部门信息的时候,级联保存该部门下的员工信息。

2、更新(Update):调用session.get()方法查询到某个部门的信息之后,调用部门对象的set()方法就可以更改部门的属性信息,再调用该部门的getEmps()方法,迭代该部门下的所有员工信息,调用员工对象的set()方法就可以更改员工的属性信息。

3、查询(Retrieve):调用session.get()方法查询到某个部门的信息之后,再调用该部门的getEmps()方法,迭代获取该部门下的所有员工信息。hibernate的查询,默认是发起两次请求,第一条sql先查询指定部门id的部门信息,第二条sql查询指定部门id的员工信息。如果想要关联查询,需要在一端的配置文件(此处是Dept.hbml.xml)中的set属性中设置fetch=join,之后,hibernate只会发起一次sql请求,查询部门表和员工表,同时两张表的部门id相同,这样,可以提高查询的效率,所以,我们一般建议设置fetch=join属性。

4、删除(Delete):调用session.get()方法查询到某个部门的信息之后,再调用session.delete()方法查询部门信息,在Dept.hbm.xml映射文件中的set属性中设置cascase=all,会同步将该部门下的所有员工的信息。如果没有设置cascade=all属性,那么删除会失败,我们必须先删除该部门下的所有员工信息,再删除该部门信息,才能成功,所以,一般建议在一端的配置文件(此处是Dept.hbm.xml)中的set属性中设置cascade=all属性,使得删除部门的时候同时删除该部门下的所有员工信息。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值