zoukankan      html  css  js  c++  java
  • 框架之 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 }
  • 相关阅读:
    DOM基础(二)
    DOM基础(一)
    JS入门(五)
    linux的用法
    一道关于运行顺序题
    vue框架
    HTML的知识点
    从队友那偷来的主席树模板(静态区间第k小)
    网络流基础模型——任务分配模型(HDU 3572)
    HDU 5521 Meeting(建图思维)
  • 原文地址:https://www.cnblogs.com/xujingyang/p/6734165.html
Copyright © 2011-2022 走看看