框架之 hibernate之关联关系映射
案例:完成CRM的联系人的保存操作
需求分析
1. 因为客户和联系人是一对多的关系,在有客户的情况下,完成联系人的添加保存操作
技术分析之Hibernate的关联关系映射之一对多映射(重点)
1. JavaWEB中一对多的设计及其建表原则 2. 先导入SQL的建表语句 * 创建数据库:create database hibernate_03; 3. 编写客户和联系人的JavaBean程序(注意一对多的编写规则) * 客户的JavaBean如下 public class Customer { private Long cust_id; private String cust_name; private Long cust_user_id; private Long cust_create_id; private String cust_source; private String cust_industry; private String cust_level; private String cust_linkman; private String cust_phone; private String cust_mobile; private Set<Linkman> linkmans = new HashSet<Linkman>(); } * 联系人的JavaBean如下 public class Linkman { private Long lkm_id; private String lkm_name; private String lkm_gender; private String lkm_phone; private String lkm_mobile; private String lkm_email; private String lkm_qq; private String lkm_position; private String lkm_memo; private Customer customer; } 4. 编写客户和联系人的映射配置文件(注意一对多的配置编写) * 客户的映射配置文件如下 <class name="com.xujingyang.domain.Customer" table="cst_customer"> <id name="cust_id" column="cust_id"> <generator class="native"/> </id> <property name="cust_name" column="cust_name"/> <property name="cust_user_id" column="cust_user_id"/> <property name="cust_create_id" column="cust_create_id"/> <property name="cust_source" column="cust_source"/> <property name="cust_industry" column="cust_industry"/> <property name="cust_level" column="cust_level"/> <property name="cust_linkman" column="cust_linkman"/> <property name="cust_phone" column="cust_phone"/> <property name="cust_mobile" column="cust_mobile"/> <set name="linkmans"> <key column="lkm_cust_id"/> <one-to-many class="com.xujingyang.domain.Linkman"/> </set> </class> * 联系人的映射配置文件如下 <class name="com.xujingyang.domain.Linkman" table="cst_linkman"> <id name="lkm_id" column="lkm_id"> <generator class="native"/> </id> <property name="lkm_name" column="lkm_name"/> <property name="lkm_gender" column="lkm_gender"/> <property name="lkm_phone" column="lkm_phone"/> <property name="lkm_mobile" column="lkm_mobile"/> <property name="lkm_email" column="lkm_email"/> <property name="lkm_qq" column="lkm_qq"/> <property name="lkm_position" column="lkm_position"/> <property name="lkm_memo" column="lkm_memo"/> <many-to-one name="customer" class="com.xujingyang.domain.Customer" column="lkm_cust_id"/> </class>
技术分析之保存客户和联系人的数据
1. 进行双向关联进行数据的保存
技术分析之级联保存
1. 测试:如果现在代码只插入其中的一方的数据
* 如果只保存其中的一方的数据,那么程序会抛出异常。
* 如果想完成只保存一方的数据,并且把相关联的数据都保存到数据库中,那么需要配置级联!!
* 级联保存是方向性
2. 级联保存效果
* 级联保存:保存一方同时可以把关联的对象也保存到数据库中!!
* 使用cascade="save-update"
技术分析之级联删除
1. 先来给大家在数据库中演示含有外键的删除客户功能,那么SQL语句是会报出错误的
* 例如:delete from customers where cid = 1;
2. 如果使用Hibernate框架直接删除客户的时候,测试发现是可以删除的
3. 上述的删除是普通的删除,那么也可以使用级联删除,注意:级联删除也是有方向性的!!
* <many-to-one cascade="delete" />
技术分析之级联的取值(cascade的取值)和孤儿删除
1. 需要大家掌握的取值如下
* none -- 不使用级联
* save-update -- 级联保存或更新
* delete -- 级联删除
* delete-orphan -- 孤儿删除.(注意:只能应用在一对多关系)
* all -- 除了delete-orphan的所有情况.(包含save-update delete)
* all-delete-orphan -- 包含了delete-orphan的所有情况.(包含save-update delete delete-orphan)
2. 孤儿删除(孤子删除),只有在一对多的环境下才有孤儿删除
* 在一对多的关系中,可以将一的一方认为是父方.将多的一方认为是子方.孤儿删除:在解除了父子关系的时候.将子方记录就直接删除。
* <many-to-one cascade="delete-orphan" />
技术分析之让某一方放弃外键的维护,为多对多做准备
1. 先测试双方都维护外键的时候,会产生多余的SQL语句。
* 想修改客户和联系人的关系,进行双向关联,双方都会维护外键,会产生多余的SQL语句。
* 产生的原因:session的一级缓存中的快照机制,会让双方都更新数据库,产生了多余的SQL语句。
2. 如果不想产生多余的SQL语句,那么需要一方来放弃外键的维护!
* 在<set>标签上配置一个inverse=”true”.true:放弃.false:不放弃.默认值是false
* <inverse="true">
技术分析之cascade和inverse的区别
1. cascade用来级联操作(保存、修改和删除)
2. inverse用来维护外键的
Hibernate的关联关系映射之多对多映射
技术分析之多对多的建表原则
1. JavaWEB的多对多
技术分析之多对多JavaBean的编写
1. 编写用户和角色的JavaBean * 用户的JavaBean代码如下 public class User { private Long user_id; private String user_code; private String user_name; private String user_password; private String user_state; private Set<Role> roles = new HashSet<Role>(); } * 角色的JavaBean代码如下 public class Role { private Long role_id; private String role_name; private String role_memo; private Set<User> users = new HashSet<User>(); } 2. 用户和角色的映射配置文件如下 * 用户的映射配置文件如下 <class name="com.xujingyang.domain.User" table="sys_user"> <id name="user_id" column="user_id"> <generator class="native"/> </id> <property name="user_code" column="user_code"/> <property name="user_name" column="user_name"/> <property name="user_password" column="user_password"/> <property name="user_state" column="user_state"/> <set name="roles" table="sys_user_role"> <key column="user_id"/> <many-to-many class="com.xujingyang.domain.Role" column="role_id"/> </set> </class> * 角色的映射配置文件如下 <class name="com.xujingyang.domain.Role" table="sys_role"> <id name="role_id" column="role_id"> <generator class="native"/> </id> <property name="role_name" column="role_name"/> <property name="role_memo" column="role_memo"/> <set name="users" table="sys_user_role"> <key column="role_id"/> <many-to-many class="com.xujingyang.domain.User" column="user_id"/> </set> </class> 3. 多对多进行双向关联的时候:必须有一方去放弃外键维护权
技术分析之多对多的级联保存
1. 级联保存
* <set cascade="save-update">
级联删除(在多对多中是很少使用的)
1. 级联删除
代码
1 package com.xujingyang.test ; 2 3 import org.hibernate.Session ; 4 import org.hibernate.Transaction ; 5 import org.junit.Test ; 6 import com.xujingyang.demo.Customer ; 7 import com.xujingyang.demo.LinkMan ; 8 import com.xujingyang.utils.HibernateUtil ; 9 10 /** 11 * @author oldmonk 12 * @date 2017年4月16日 13 */ 14 public class TestDemo { 15 16 /** 17 * 双向添加 18 */ 19 @Test 20 public void Demo1() { 21 Session session = HibernateUtil.getThreadLocalSession() ; 22 Transaction transaction = session.beginTransaction() ; 23 24 Customer customer = new Customer() ; 25 customer.setCust_name("小明") ; 26 27 LinkMan lMan1 = new LinkMan() ; 28 lMan1.setLkm_name("小小明1") ; 29 LinkMan lMan2 = new LinkMan() ; 30 lMan2.setLkm_name("小小明2") ; 31 32 customer.getLinkmans().add(lMan1) ; 33 customer.getLinkmans().add(lMan2) ; 34 lMan1.setCustomer(customer) ; 35 lMan2.setCustomer(customer) ; 36 37 session.save(customer) ; 38 session.save(lMan1) ; 39 session.save(lMan2) ; 40 41 transaction.commit() ; 42 43 } 44 45 /** 46 * 级联添加, 客户配置中添加cascade="all"此时还维护外建,会执行两次重复的sql语句 , 47 * 添加 inverse="true"时,表明客户表不维护外键,不会重复执行 48 */ 49 @Test 50 public void Demo2() { 51 Session session = HibernateUtil.getThreadLocalSession() ; 52 Transaction transaction = session.beginTransaction() ; 53 54 Customer customer = new Customer() ; 55 customer.setCust_name("小明") ; 56 57 LinkMan lMan1 = new LinkMan() ; 58 lMan1.setLkm_name("小小明1") ; 59 LinkMan lMan2 = new LinkMan() ; 60 lMan2.setLkm_name("小小明2") ; 61 62 customer.getLinkmans().add(lMan1) ; 63 customer.getLinkmans().add(lMan2) ; 64 65 session.save(customer) ; 66 67 transaction.commit() ; 68 69 } 70 71 /** 72 * 普通删除,此时会把关联的外建表的列对应的外建置为null 73 * 不添加任何的delete 74 * 删除客户的时候 删除联系人: 75 * 需要在Customer.hbm.xml中set标签上配置cascade="delete" 76 * 删除联系人 同时删除 客户: 77 * 需要在LinkMan.hbm.xml中many-to-one标签上配置cascade="delete" 78 */ 79 @Test 80 public void Demo3() { 81 Session session = HibernateUtil.getThreadLocalSession() ; 82 Transaction transaction = session.beginTransaction() ; 83 84 Customer customer = session.get(Customer.class, 1L) ; 85 86 session.delete(customer) ; 87 88 transaction.commit() ; 89 } 90 91 /** 92 * 孤儿删除, <set cascade="delete-orphan" />,跟我解除关系,就把你删了 93 * 不添加时,只是解除关系,外建置为null 94 */ 95 @Test 96 public void Demo4() { 97 Session session = HibernateUtil.getThreadLocalSession() ; 98 Transaction transaction = session.beginTransaction() ; 99 100 Customer customer = session.get(Customer.class, 1L) ; 101 102 LinkMan linkMan = session.get(LinkMan.class, 1L) ; 103 customer.getLinkmans().remove(linkMan) ; 104 105 transaction.commit() ; 106 } 107 }