美文网首页
hibernate 一对多、多对多操作

hibernate 一对多、多对多操作

作者: 夏天小哥哥 | 来源:发表于2019-11-25 22:00 被阅读0次

表之间的关系

  1. 一对多

    一个部门有多个员工,一个员工只能属于某一个部门
    一个班级有多个学生,一个学生只能属于一个班级

  2. 多对多

    一个老师教多个学生,一个学生可以被多个老师教
    一个学生可以先择多门课程,一门课程可以被多个学生选择
    一个用户可以选择多个角色,一个角色也可以被多个用户选择

  3. 一对一

    一个公司只能对应一个注册地址

表之间关系建表原则

  1. 一对多

    在多的一方创建一个外键,指向一的一方的主键

  2. 多对多

    创建一个中间表,中间表至少有两个字段,分别作为外键指向多对多双方的主键

  3. 一对一

    唯一外键对应或主键对应

多表操作

一对多
  1. 建立表

    CREATE TABLE `linkman` (
      `link_id` bigint(32) NOT NULL AUTO_INCREMENT COMMENT '联系人编号(主键)',
      `link_name` varchar(16) DEFAULT NULL COMMENT '联系人姓名',
      `link_cust_id` bigint(32) NOT NULL COMMENT '客户id',
      `link_gender` char(1) DEFAULT NULL COMMENT '联系人性别',
      `link_phone` varchar(16) DEFAULT NULL COMMENT '联系人办公电话',
      `link_mobile` varchar(16) DEFAULT NULL COMMENT '联系人手机',
      `link_email` varchar(64) DEFAULT NULL COMMENT '联系人邮箱',
      `link_qq` varchar(16) DEFAULT NULL COMMENT '联系人qq',
      `link_position` varchar(16) DEFAULT NULL COMMENT '联系人职位',
      `link_memo` varchar(512) DEFAULT NULL COMMENT '联系人备注',
      PRIMARY KEY (`link_id`)
    ) comment '联系人表' ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;
    
    CREATE TABLE `customer` (
      `cust_id` bigint(32) NOT NULL AUTO_INCREMENT COMMENT '客户编号(主键)',
      `cust_name` varchar(32) NOT NULL COMMENT '客户名称(公司名称)',
      `cust_source` varchar(32) DEFAULT NULL COMMENT '客户信息来源',
      `cust_industry` varchar(32) DEFAULT NULL COMMENT '客户所属行业',
      `cust_level` varchar(32) DEFAULT NULL COMMENT '客户级别',
      `cust_phone` varchar(64) DEFAULT NULL COMMENT '固定电话',
      `cust_mobile` varchar(16) DEFAULT NULL COMMENT '移动电话',
      PRIMARY KEY (`cust_id`)
    ) comment '客户表' ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
    
  2. 创建实体类并设置表之间的关系

    /**
     * 客户实体类
     * @author zhou
     * @create 2019/11/19/21:33
     * @class com.zhou.domian.Linkman
     */
    @Getter
    @Setter
    public class Customer {
    
        private long cust_id;
        private String cust_name;
        private String cust_source;
        private String cust_industry;
        private String cust_level;
        private String cust_phone;
        private String cust_mobile;
    
        // 一个客户有多个联系人
        private Set<Linkman> linkmens = new HashSet<>();
    
    }
    
    /**
     * 联系人实体类
     * @author zhou
     * @create 2019/11/19/21:11
     * @class com.zhou.domian.Linkman
     */
    @Getter
    @Setter
    public class Linkman {
        private Long link_id;
        private String link_name;
        private String link_gender;
        private String link_phone;
        private String link_mobile;
        private String link_email;
        private String link_qq;
        private String link_position;
        private String link_memo;
    
        // 联系人表关联一个客户
        private Customer customer;
    }
    
  3. 创建映射文件

    <?xml version='1.0' encoding='utf-8'?>
    <!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="com.zhou.domian.Customer" table="customer" schema="hibernate">
        <!-- 建立那个字段是主键 -->
        <id name="cust_id" column="cust_id" >
            <!-- 主键的生成策略 -->
            <generator class="native"/>
        </id>
        <!-- 建立POJO类字段和数据库字段关联 -->
        <property name="cust_name" column="cust_name"/>
        <property name="cust_source" column="cust_source"/>
        <property name="cust_industry" column="cust_industry"/>
        <property name="cust_level" column="cust_level"/>
        <property name="cust_phone" column="cust_phone"/>
        <property name="cust_mobile" column="cust_mobile"/>
    
        <!-- set标签的name属性: 多的一方的集合的属性名称  -->
        <set name="linkmens">
            <!-- key的 column表示多的一方外键名 -->
            <key column="link_cust_id"></key>
            <!-- one-to-many的class属性表示多的一方类的全限定名 -->
            <one-to-many class="com.zhou.domian.Linkman"></one-to-many>
        </set>
    </class>
    
    </hibernate-mapping>
    
    <?xml version='1.0' encoding='utf-8'?>
    <!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="com.zhou.domian.Linkman" table="linkman" schema="hibernate">
            <id name="link_id" column="link_id">
                <generator class="native"></generator>
            </id>
            <property name="link_name" column="link_name"/>
            <property name="link_gender" column="link_gender"/>
            <property name="link_phone" column="link_phone"/>
            <property name="link_mobile" column="link_mobile"/>
            <property name="link_email" column="link_email"/>
            <property name="link_qq" column="link_qq"/>
            <property name="link_position" column="link_position"/>
            <property name="link_memo" column="link_memo"/>
    
            <!--多对一
                name: 一的一方对象的名字
                class: 一的一方类的全限定名
                column: 多的一方外键ID的名称
            -->
            <many-to-one name="customer" column="link_cust_id" class="com.zhou.domian.Customer" />
        </class>
    
    </hibernate-mapping>
    
  4. 创建核心配置文件

    <!DOCTYPE hibernate-configuration PUBLIC
            "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
            "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
    
    <hibernate-configuration>
        <session-factory>
            <!-- 设置连接数据库的url和用户名和密码和数据库驱动-->
            <property name="url">jdbc:mysql:///hibernate</property>
            <property name="username">root</property>
            <property name="password">123456</property>
            <property name="driverClass">com.mysql.jdbc.Driver</property>
            <!-- 打印SQL语句 -->
            <property name="hibernate.show_sql">true</property>
            <!-- 格式化SQL -->
            <property name="hibernate.format_sql">true</property>
            <!-- 设置hibernate的方言 -->
            <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</property>
            <!-- 设置自动创建表 -->
            <property name="hibernate.hbm2ddl.auto">create</property>
    
            <!--阿里巴巴 Druid 连接池 注意:如果使用Druid连接池的话需要把数据基本连接属性改为Druid的属性-->
            <property name="hibernate.connection.provider_class">
                com.alibaba.druid.support.hibernate.DruidConnectionProvider
            </property>
    
            <!-- 配置初始化大小、最小、最大 -->
            <property name="initialSize">1</property>
            <property name="minIdle">5</property>
            <property name="maxActive">20</property>
            <!-- 配置获取连接等待超时的时间 -->
            <property name="maxWait">60000</property>
            <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
            <property name="timeBetweenEvictionRunsMillis">60000</property>
            <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
            <property name="minEvictableIdleTimeMillis">300000</property>
    
            <!-- 设置事务隔离级别 -->
            <property name="hibernate.connection.isolation">3</property>
    
            <!-- 创建一个session绑定到当前线程 -->
            <property name="current_session_context_class">thread</property>
    
            <!-- 加载映射(mapper)文件 -->
            <mapping resource="mapper/customer.hbm.xml"></mapping>
            <mapping resource="mapper/linkman.hbm.xml"/>
        </session-factory>
    </hibernate-configuration>
    
  5. 引入工具类

    package com.zhou.utils;
    
    import org.hibernate.Session;
    import org.hibernate.SessionFactory;
    import org.hibernate.cfg.Configuration;
    
    /**
     * @author zhou
     * @create 2019/11/12/21:21
     * @class com.zhou.utils.HibernateUtils
     */
    public class HibernateUtils {
    
        public static final SessionFactory sessionFactory;
    
        static {
            // 读取配置,并创建工厂类SessionFactory
            sessionFactory = new Configuration().configure().buildSessionFactory();
        }
    
        /**
         * 获取Session对象
         * @return Session
         */
        public static Session getSession () {
            return sessionFactory.openSession();
        }
    
        /**
         * 使用时必须要在hibernate核心配置文件中配置
         * <property name="current_session_context_class">thread</property>
         * 获取Session对象,不过是从内部已经绑定好了ThreadLocal,获取的Session对象
         * @return Session
         */
        public static Session getCurrentSession(){
            return sessionFactory.getCurrentSession();
        }
    
    }
    
  6. 编写测试类

    级联操作与懒加载

    注意: 谁需要级联操作,就在mapper(映射文件)中加cascade属性


    级联的配置属性

    6.1 级联查询与懒加载

    <many-to-one name="customer" column="link_cust_id" class="com.zhou.domian.Customer" lazy="proxy" />
    
       /**
        * @author zhou
        * @create 2019/11/19/23:15
        * @class com.zhou.test.HibernateORMTest
        */
       public class HibernateORMTest {
       
           /**
            * 级联查询与懒加载
            */
           @Test
           public void get(){
               Session session = HibernateUtils.getCurrentSession();
               Transaction transaction = session.beginTransaction();
               Linkman linkman = session.get(Linkman.class, 1L);
               transaction.commit();
       
               System.out.println(linkman.getLink_name());
               /**
                * org.hibernate.LazyInitializationException: could not initialize proxy [com.zhou.domian.Customer#1] - no Session
                * 报这个错误是因为使用hibernate默认使用了懒加载的方式,要使用到另外一个对象时才会发送SQL去查询,
                * 而getCurrentSession()方法取出来的Session在commit后,会自动把session.close()掉,所以报no session的错误
                * 如果不想要对象懒加载可以在mapper配置文件中配置,lazy: 设置是否懒加载,默认为proxy(懒加载),false为不启用懒加载
                * <many-to-one name="customer" column="link_cust_id" class="com.zhou.domian.Customer" lazy="proxy" />
                */
               System.out.println(linkman.getCustomer().getCust_name());
           }
       }
    

    6.2. 级联保存

       <set name="linkmens" cascade="save-update" lazy="false">
               <!-- key的 column表示多的一方外键名 -->
               <key column="link_cust_id"></key>
               <!-- one-to-many的class属性表示多的一方类的全限定名 -->
               <one-to-many class="com.zhou.domian.Linkman"></one-to-many>
       </set> 
    
      /**
         * @author zhou
         * @create 2019/11/19/23:15
         * @class com.zhou.test.HibernateORMTest
         */
        public class HibernateORMTest {
        
            /**
             * 一对多的级联保存save, 双向绑定,不需要设置cascade属性
             */
            @Test
            public void save () {
        
                // 获取Session对象并开启事务
                Session session = HibernateUtils.getCurrentSession();
                Transaction transaction = session.beginTransaction();
        
                // 创建实体类并设置关系
                Customer customer1 = new Customer();
                Customer customer2 = new Customer();
                Customer customer3 = new Customer();
        
                customer1.setCust_name("customer1");
                customer2.setCust_name("customer2");
                customer3.setCust_name("customer3");
        
                Linkman linkman1 = new Linkman();
                Linkman linkman2 = new Linkman();
                Linkman linkman3 = new Linkman();
        
                linkman1.setLink_name("linkman1");
                linkman2.setLink_name("linkman2");
                linkman3.setLink_name("linkman3");
        
        
                linkman1.setCustomer(customer1);
                linkman2.setCustomer(customer1);
                linkman3.setCustomer(customer3);
        
                customer1.getLinkmens().add(linkman1);
                customer1.getLinkmens().add(linkman2);
                customer2.getLinkmens().add(linkman3);
        
                // 保存数据
                session.save(customer1);
                session.save(customer2);
                session.save(customer3);
        
                session.save(linkman1);
                session.save(linkman2);
                session.save(linkman3);
        
                // 提交事务
                transaction.commit();
        
            }
        
            /**
             * 一对多的级联保存save, 单项绑定
             */
            @Test
            public void saveOne () {
        
                // 获取Session对象并开启事务
                Session session = HibernateUtils.getCurrentSession();
                Transaction transaction = session.beginTransaction();
        
                // 创建实体类并设置关系,单项绑定
                Customer customer1 = new Customer();
                Customer customer2 = new Customer();
                Customer customer3 = new Customer();
        
                customer1.setCust_name("customer1");
                customer2.setCust_name("customer2");
                customer3.setCust_name("customer3");
        
                Linkman linkman1 = new Linkman();
                Linkman linkman2 = new Linkman();
                Linkman linkman3 = new Linkman();
        
                linkman1.setLink_name("linkman1");
                linkman2.setLink_name("linkman2");
                linkman3.setLink_name("linkman3");
        
                /**
                 * java.lang.IllegalStateException: org.hibernate.TransientObjectException: object references an unsaved transient instance -
                 * save the transient instance before flushing: com.zhou.domian.Linkman
                 * 持久态对象关联了一个瞬时态对象,就会报瞬时对象异常
                 * 如果想要支持单项绑定,在mapper配置文件中添加cascade="save-update"
                 * set标签的name属性: 多的一方的集合的属性名称
                 * <set name="linkmens" cascade="save-update">
                 *      key的 column表示多的一方外键名
                 *      <key column="link_cust_id"></key>
                 *       one-to-many的class属性表示多的一方类的全限定名
                 *      <one-to-many class="com.zhou.domian.Linkman"></one-to-many>
                 * </set>
                 *
                 */
                customer1.getLinkmens().add(linkman1);
                customer1.getLinkmens().add(linkman2);
                customer2.getLinkmens().add(linkman3);
        
                // 保存数据
                session.save(customer1);
                session.save(customer2);
                session.save(customer3);
        
                // 提交事务
                transaction.commit();
        
            }
        }  
    
    

    6.3. 级联删除

           <set name="linkmens" cascade="delete" lazy="false">
               <!-- key的 column表示多的一方外键名 -->
               <key column="link_cust_id"></key>
               <!-- one-to-many的class属性表示多的一方类的全限定名 -->
               <one-to-many class="com.zhou.domian.Linkman"></one-to-many>
           </set>
    
       public class HibernateORMTest {
    
           /**
            * 级联删除
            */
           @Test
           public void delete() {
               // 获取Session对象并开启事务
               Session session = HibernateUtils.getCurrentSession();
               Transaction transaction = session.beginTransaction();
               // 进行查询并删除,如果没有设置级联属性则不会级联删除,只会把附属表的属性先更新为null,在删除主表中的数据
               Customer customer = session.get(Customer.class, 1L);
               session.delete(customer);
               // 提交事务
               transaction.commit();
           }
       }
    

    6.4. 外键维护权inverse

       <!-- inverse="true" 使对象放弃外键维护-->
       <!-- set标签的name属性: 多的一方的集合的属性名称  -->
       <set name="linkmens" cascade="save-update, delete" inverse="true" lazy="false">
           <!-- key的 column表示多的一方外键名 -->
           <key column="link_cust_id"></key>
           <!-- one-to-many的class属性表示多的一方类的全限定名 -->
           <one-to-many class="com.zhou.domian.Linkman"></one-to-many>
       </set>
    
     public class HibernateORMTest{
          /**
           * 级联更新外键
           * inverse外加维护权,双向绑定使用
           */
          @Test
          public void updateInverse() {
              // 获取Session对象并开启事务
              Session session = HibernateUtils.getCurrentSession();
              Transaction transaction = session.beginTransaction();
              Customer customer2 = session.get(Customer.class, 2L);
              Linkman linkman2 = session.get(Linkman.class, 2L);
              // 双向绑定,如果双向绑定,hibernate会发送两个sql来更改同一个数据,这就造成数据库的压力,需要进行优化
              // 如果想要优化,那就让一方放弃外键维护,需要设置inverse属性让一方放弃,默认是false,false是不放弃, true是放弃
              customer2.getLinkmens().add(linkman2);
              linkman2.setCustomer(customer2);
              // 提交事务
              transaction.commit();
      
          }
      
          /**
           * 级联更新外键
           * 单项绑定
           */
          @Test
          public void update() {
              // 获取Session对象并开启事务
              Session session = HibernateUtils.getCurrentSession();
              Transaction transaction = session.beginTransaction();
              Customer customer2 = session.get(Customer.class, 2L);
              Linkman linkman2 = session.get(Linkman.class, 2L);
              // 单项绑定, 只需要一个对象进行绑定.
              customer2.getLinkmens().add(linkman2);
              // linkman2.setCustomer(customer2);
              // 提交事务
              transaction.commit();
          }
     }
    
多对多
  1. 建立表

       # 用户表
           #一个用户可以有多个角色
            CREATE TABLE `user` (
                  `user_id` bigint(20) NOT NULL AUTO_INCREMENT,
                  `user_code` varchar(255) DEFAULT NULL,
                  `user_name` varchar(255) DEFAULT NULL,
                  `user_password` varchar(255) DEFAULT NULL,
                  `user_state` varchar(255) DEFAULT NULL,
                  PRIMARY KEY (`user_id`)
                ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
        #角色表
           #一个角色可以被多个用户选择
            CREATE TABLE `role` (
                  `role_id` bigint(20) NOT NULL AUTO_INCREMENT,
                  `role_name` varchar(255) DEFAULT NULL,
                  `role_memo` varchar(255) DEFAULT NULL,
                  PRIMARY KEY (`role_id`)
                ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
       # 中间表
               CREATE TABLE `user_role` (
                 `role_id` bigint(20) NOT NULL,
                 `user_id` bigint(20) NOT NULL,
                 PRIMARY KEY (`user_id`,`role_id`),
                 CONSTRAINT `user_id_fk` FOREIGN KEY (`user_id`) REFERENCES `user` (`user_id`),
                 CONSTRAINT `role_id_fk` FOREIGN KEY (`role_id`) REFERENCES `role` (`role_id`)
               ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; 
    

    关系图


    关系图.png
  2. 创建实体类并建立关系

       /**
        * @author zhou
        * @create 2019/11/24/19:33
        * @class com.zhou.domian.User
        */
       @Getter
       @Setter
       public class User {
           private long user_id;
           private String user_code;
           private String user_name;
           private String user_password;
           private String user_state;
       
           /**
            * 建立用户对应多个角色的映射
            */
           Set<Role> roles = new HashSet<>();
       }
    
       /**
        * @author zhou
        * @create 2019/11/24/19:33
        * @class com.zhou.domian.Role
        */
       @Getter
       @Setter
       public class Role {
           private long role_id;
           private String role_name;
           private String role_memo;
       
           /**
            * 建立多角色对应多用户
            */
           private Set<User> users = new HashSet<>();
       }
    
  3. 添加配置文件

    <?xml version='1.0' encoding='utf-8'?>
      <!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="com.zhou.domian.User" table="user" schema="hibernate">
              <id name="user_id" column="user_id">
                  <generator class="native"></generator>
              </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"/>
      
      
              <!--
                  name: 是当前集合属性名称
                  table: 是多对多中间表
                  <key column="role_id"></key>
                  当前表的外键
                  <many-to-many column="role_id" class="com.zhou.domian.Role"/>
                  column: 集合中对象的外键
                  class: 集合中对象的全路径
              -->
              <set name="roles" table="user_role" cascade ="save-update">
                  <key column="user_id"></key>
                  <many-to-many class="com.zhou.domian.Role" column="role_id"/>
              </set>
          </class>
    </hibernate-mapping>  
    
    <?xml version='1.0' encoding='utf-8'?>
        <!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="com.zhou.domian.Role" table="role" schema="hibernate">
                <!-- 建立那个字段是主键 -->
                <id name="role_id" column="role_id">
                    <!-- 主键的生成策略 -->
                    <generator class="native"/>
                </id>
                <!-- 建立POJO类字段和数据库字段关联 -->
                <property name="role_name" column="role_name"/>
                <property name="role_memo" column="role_memo"/>
                <!--
                    name: 是当前集合属性名称
                    table: 是多对多中间表
                    <key column=""></key>
                    当前表的外键
                    <many-to-many column="" class=""/>
                    column: 集合中对象的外键
                    class: 集合中对象的全路径
                -->
                <set name="users" table="user_role" inverse="false">
                    <key column="role_id"></key>
                    <many-to-many column="user_id" class="com.zhou.domian.User"/>
                </set>
            </class>
    </hibernate-mapping>
    
  4. 在核心配置文件当中添加两个新配置

    <!DOCTYPE hibernate-configuration PUBLIC
            "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
            "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
    
    <hibernate-configuration>
        <session-factory>
            <!-- 设置连接数据库的url和用户名和密码和数据库驱动-->
            <property name="url">jdbc:mysql:///hibernate</property>
            <property name="username">root</property>
            <property name="password">123456</property>
            <property name="driverClass">com.mysql.jdbc.Driver</property>
            <!-- 打印SQL语句 -->
            <property name="hibernate.show_sql">true</property>
            <!-- 格式化SQL -->
            <property name="hibernate.format_sql">true</property>
            <!-- 设置hibernate的方言 -->
            <property name="hibernate.dialect">org.hibernate.dialect.MySQL57Dialect</property>
            <!-- 设置自动创建表 -->
            <property name="hibernate.hbm2ddl.auto">update</property>
    
            <!--阿里巴巴 Druid 连接池 注意:如果使用Druid连接池的话需要把数据基本连接属性改为Druid的属性-->
            <property name="hibernate.connection.provider_class">
                com.alibaba.druid.support.hibernate.DruidConnectionProvider
            </property>
    
            <!-- 配置初始化大小、最小、最大 -->
            <property name="initialSize">1</property>
            <property name="minIdle">5</property>
            <property name="maxActive">20</property>
            <!-- 配置获取连接等待超时的时间 -->
            <property name="maxWait">60000</property>
            <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
            <property name="timeBetweenEvictionRunsMillis">60000</property>
            <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
            <property name="minEvictableIdleTimeMillis">300000</property>
    
            <!-- 设置事务隔离级别 -->
            <property name="hibernate.connection.isolation">3</property>
    
            <!-- 创建一个session绑定到当前线程 -->
            <property name="current_session_context_class">thread</property>
    
            <!-- 加载映射(mapper)文件 -->
            <!--一对多的映射文件-->
            <mapping resource="mapper/customer.hbm.xml"/>
            <mapping resource="mapper/linkman.hbm.xml"/>
            <!--添加多对多的映射文件-->
            <mapping resource="mapper/role.hbm.xml"/>
            <mapping resource="mapper/user.hbm.xml"/>
        </session-factory>
    </hibernate-configuration>
    
    
  5. 编写测试类

    public class HibernateManyAndMany {
    
       /**
        * 双向维护
        * 级联保存,双向维护,从表开启放弃外键维护, 如在mapper映射文件中的多对多属性中加上inverse="false",放弃外键维
        * 双向维护时,必须要有一方放弃外键维护
        * 如果两边都有维护的话, 就会有重复的的记录,由于关系表是两个字段作为共同主键,不能有相同的记录
        * 解决办法
        * 通常都是让被动方放弃,用户选角色,角色为被动方
        */
        @Test
        public void save() {
            // 获取Session对象并开启事务
            Session currentSession = HibernateUtils.getCurrentSession();
            Transaction transaction = currentSession.beginTransaction();
    
            User user1 = new User();
            user1.setUser_name("user1");
            User user2 = new User();
            user2.setUser_name("user2");
            User user3 = new User();
            user3.setUser_name("user3");
    
            Role role1 = new Role();
            role1.setRole_name("施工员");
            Role role2 = new Role();
            role2.setRole_name("资料员");
            Role role3 = new Role();
            role3.setRole_name("项目经理");
    
            user1.getRoles().add(role1);
            user1.getRoles().add(role3);
            user2.getRoles().add(role1);
            user3.getRoles().add(role2);
    
            currentSession.save(user1);
            currentSession.save(user2);
            currentSession.save(user3);
            // 放弃外键维护
            currentSession.save(role1);
            currentSession.save(role2);
            currentSession.save(role3);
    
            // 提交事务
            transaction.commit();
        }
    }
    
    /**
    * @author zhou
    * @create 2019/11/24/20:01
    * @class com.zhou.test.HibarnateManyAndMany
    */
    public class HibernateManyAndMany {
       /**
        * 级联保存,单项绑定, 单项绑定需要配置cascade="save-update"
        */
       @Test
       public void saveOne() {
           // 获取Session对象并开启事务
           Session currentSession = HibernateUtils.getCurrentSession();
           Transaction transaction = currentSession.beginTransaction();
    
           User user1 = new User();
           user1.setUser_name("user1");
           User user2 = new User();
           user2.setUser_name("user2");
           User user3 = new User();
           user3.setUser_name("user3");
    
           Role role1 = new Role();
           role1.setRole_name("施工员");
           Role role2 = new Role();
           role2.setRole_name("资料员");
           Role role3 = new Role();
           role3.setRole_name("项目经理");
    
           user1.getRoles().add(role1);
           user1.getRoles().add(role3);
           user2.getRoles().add(role1);
           user3.getRoles().add(role2);
    
           currentSession.save(user1);
           currentSession.save(user2);
           currentSession.save(user3);
    
           // 提交事务
           transaction.commit();
       }
    }
    

    多对多的级联操作和一对多的级联操作是一样的

    /**
     * @author zhou
     * @create 2019/11/24/20:01
     * @class com.zhou.test.HibarnateManyAndMany
     */
    public class HibernateManyAndMany {
    
        /**
         * 添加角色,双向绑定,让一方放弃外键维护,需要设置inverse属性让一方放弃,默认是false,false是不放弃, true是放弃
         * 如果不让一方放弃外键维护的话,会报错
         */
        @Test
        public void addRole() {
            // 获取Session并开启事务
            Session session = HibernateUtils.getCurrentSession();
            Transaction transaction = session.beginTransaction();
    
            // 用户3添加角色1
            User user3 = session.get(User.class, 3L);
            Role role1 = session.get(Role.class, 1L);
            user3.getRoles().add(role1);
            role1.getUsers().add(user3);
            // 提交事务
            transaction.commit();
        }
    
    
        /**
         * 单项维护,添加角色
         */
        @Test
        public void addRoleOne() {
            // 获取Session并开启事务
            Session session = HibernateUtils.getCurrentSession();
            Transaction transaction = session.beginTransaction();
    
            // 用户3添加角色1
            User user3 = session.get(User.class, 3L);
            Role role1 = session.get(Role.class, 1L);
            user3.getRoles().add(role1);
            // 提交事务
            transaction.commit();
        }
    
    
        /**
         * 关系操作,就是操作内部的集合
         * 把用户1的角色3修改为角色2
         */
        @Test
        public void UpdateUserAndRole() {
            // 获取Session并开启事务
            Session session = HibernateUtils.getCurrentSession();
            Transaction transaction = session.beginTransaction();
    
            // 把用户1的角色3修改为角色2
            User user1 = session.get(User.class, 1L);
            Role role2 = session.get(Role.class, 2L);
            Role role3 = session.get(Role.class, 3L);
            // 删除角色2
            user1.getRoles().remove(role3);
            // 添加角色3
            user1.getRoles().add(role2);
    
            // 提交事务
            transaction.commit();
        }
    }
    

相关文章

  • hibernate 一对多、多对多操作

    表之间的关系 一对多一个部门有多个员工,一个员工只能属于某一个部门一个班级有多个学生,一个学生只能属于一个班级 多...

  • Hibernate 一对多操作

    一 表与表之间关系回顾 一对多 多对多 一对一 二 hibernate一对多操作1 一对多映射配置以客户联系人为列...

  • Hibernate 多对多操作

    一 多对多映射配置1 创建实体类,用户和角色UserDemo.java Role.java 2 两个实体类之间互相...

  • Java:hibernate框架(3)

    一. 表与表之间的关系 二.Hibernate的一对多操作1.一对多映射配置 2.一对多级联保存 3.一对多的级联...

  • hibernate 一对多

    1.0 最基本的双向一对多 1.1 添加 1.2 修改 1.3 删除 执行后报错,这里原因是child那里我们设置...

  • Hibernate学习(Hibernate的一对多操作)

    一对多映射配置(重点) 以客户和联系人为例:客户是一,联系人是多 第一步 创建两个实体类,客户和联系人 第二步 让...

  • hibernate里的一对多,多对一,多对多

  • Hibernate学习(多对多的操作)

    多对多映射配置 以用户和角色为例演示、第一步 创建实体类,用户和角色第二步 让两个实体类之间互相表示(1)一个用户...

  • hibernate 多对多

    0.0 参数 class.catalog="指定数据库的,建议删掉" 1.0 简单多对多(双向多对多) 1.1添加...

  • 2018-11-18

    学习hibernate框架中的数据库,一对多或者多对一,多对多,实体类之间的关系

网友评论

      本文标题:hibernate 一对多、多对多操作

      本文链接:https://www.haomeiwen.com/subject/rdkswctx.html