Hibernate深入理解----04关联关系(单向多对1,双向1对多)

本文介绍了Hibernate中单向多对一和双向一对多关联关系的映射方式,包括类的设计、XML映射文件的配置及如何通过代码实现保存、获取、更新和删除等操作。

参考代码下载github:https://github.com/changwensir/java-ee/tree/master/hibernate4

在领域模型中 , 类与类之间最普遍的关系就是关联关系 .在UML图中,关联是有方向的

1.单向多对1

两个类,多个Order对应一个Customer

public class Customer {
    private Integer customerId;
    private String customerName;

//省去get,set方法
}

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<hibernate-mapping package="com.changwen.hibernate4.mapped.manyToOne.pojo">
    
    <class name="Customer" table="customers">
    
        <id name="customerId" column="customer_id" type="java.lang.Integer">
            <generator class="native" />
        </id>
    
        <property name="customerName" column="customer_name" type="java.lang.String"/>
    </class>
    
</hibernate-mapping>
Order类
public class Order {
    private Integer orderId;
    private String orderName;
    private Customer customer;

//省去get,set方法
}
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<hibernate-mapping package="com.changwen.hibernate4.mapped.manyToOne.pojo">

    <class name="Order" table="orders">

        <id name="orderId" type="java.lang.Integer">
            <column name="order_id" />
            <generator class="native" />
        </id>
        
        <property name="orderName" type="java.lang.String">
            <column name="order_name" />
        </property>

        <!--多个Order对应一个Customer

           映射多对一的关联关系。 使用 many-to-one 来映射多对一的关联关系
           name: 多对一端关联的一那端的属性的名字
           class: 一那端的属性对应的类名
           column: 一那端在多的一端对应的数据表中的外键的名字,随意取
       -->
        <many-to-one name="customer" class="Customer" column="customer_ID"/>

    </class>
</hibernate-mapping>
然后把两个hbm.xml映射到cfg.xml里
测试:before,after的见前面的内容
    @Test
    public void testMany2OneSave(){
        Customer customer = new Customer();
        customer.setCustomerName("BB");

        Order order1 = new Order();
        order1.setOrderName("ORDER-3");
        Order order2 = new Order();
        order2.setOrderName("ORDER-4");

        //设定关联关系
        order1.setCustomer(customer);
        order2.setCustomer(customer);

        //这种写法更好
        //执行  save 操作: 先插入 Customer, 再插入 Order, 3 条 INSERT
        //先插入 1 的一端, 再插入 n 的一端, 只有 INSERT 语句.
        //推荐先插入 1 的一端, 后插入 n 的一端
	session.save(customer);
	session.save(order1);
	session.save(order2);

        //先插入 Order, 再插入 Customer. 3 条 INSERT, 2 条 UPDATE
        //先插入 n 的一端, 再插入 1 的一端, 会多出 UPDATE 语句!
        //因为在插入多的一端时, 无法确定 1 的一端的外键值. 所以只能等 1 的一端插入后, 再额外发送 UPDATE 语句.
/*        sessionPojo.save(order1);
        sessionPojo.save(order2);
        sessionPojo.save(customer);*/
    }

    @Test
    public void testMany2OneGet(){
        //结论1. 若查询多的一端的一个对象, 则默认情况下, 只查询了多的一端的对象.
        //      而没有查询关联的1 的那一端的对象!
//        Order order = (Order) sessionPojo.get(Order.class, 8);
//        System.out.println(order.getOrderName());

        //结论2. 在需要使用到关联的对象时, 才发送对应的 SQL 语句.
//        Customer customer = order.getCustomer();
//        System.out.println(customer.getCustomerName());
        //上面1,2的输出结果顺序是:sql语句-->sout-->sql语句-->sout

        //3. 在查询 Customer 对象时, 由多的一端导航到 1 的一端时,
        //若此时 sessionPojo 已被关闭, 则默认情况下会发生 LazyInitializationException 异常
//        Order order2 = (Order) session.get(Order.class, 8);
//        System.out.println(order2.getOrderName());

//        session.close(); //这里测试时需要把destory()方法里的transaction和session注解掉

//        Customer customer2 = order2.getCustomer();
//        System.out.println(customer2.getCustomerName());
        //上面3的输出结果顺序是:sql语句-->order2.getOrderName()-->异常

        //4. 获取 Order 对象时, 默认情况下, 其关联的 Customer 对象是一个代理对象!
        Order order3 = (Order) session.get(Order.class, 8);
        System.out.println(order3.getCustomer().getClass().getName());
    }
    @Test
    public void testDelete(){
        //在不设定级联关系的情况下, 且 1 这一端的对象有 n 的对象在引用, 不能直接删除 1 这一端的对象
        Customer customer = (Customer) session.get(Customer.class, 8);
        session.delete(customer); //不能删除
    }

    @Test
    public void testUpdate(){
        Order order = (Order) session.get(Order.class, 8);
        order.getCustomer().setCustomerName("AAA"); //有对应的update的语句
    }

2、双向1对多

Order类与上面的配置一样,Customer类修改如下

public class Customer {
    private Integer customerId;
    private String customerName;
    /**
     * 1.声明集合类型时,需使用接口类型。因为hibernate 在获取集合类型时,
     *   返回的是Hibernate 内置的集合类型,而不是JavaSE 一个标准集合实现
     * 2.需要把集合进行初始化,可以防止空指针异常
     */
    private Set<Order> orders = new HashSet<Order>();

    public Set<Order> getOrders() {
        return orders;
    }

    public void setOrders(Set<Order> orders) {
        this.orders = orders;
    }
//省去get,set方法
}
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">

<hibernate-mapping package="com.changwen.hibernate4.mapped.oneToManyBoth.pojo">
    
    <class name="Customer" table="customers">
    
        <id name="customerId" column="customer_id" type="java.lang.Integer">
            <generator class="native" />
        </id>

        <property name="customerName" column="customer_name" type="java.lang.String"/>

        <!--
            映射1对多的那个集合属性
            name是Customer类里属性,table是Order.hbm.xml里的<class name="Order" table="orders">的table里的值
            外键column跟Order.hbm.xml里 <many-to-one里的column里值一样\

            inverse 属性的来决定是由双向关联的哪一方来维护表和表之间的关系
            cascade 设置级联属性,开发时不建议设置该属性,建议使用手工的方式来处理
            order-by 在查询时对集合的元素进行排序,order-by 中使用的是表的字段名,而不是类的属性名
        -->
        <set name="orders" table="orders" inverse="true" cascade="delete">
            <key column="customer_ID2"/>
            <one-to-many class="Order"/>
        </set>
    </class>
    
</hibernate-mapping>

把hbm.xml配置到cfg.xml里

    /**
     * 在hibernate.hbm.xml中通过对 inverse 属性的来决定是由双向关联的哪一方来维护表和表之间的关系.
     * inverse = false 的为主动方,inverse = true 的为被动方, 由主动方负责维护关联关系
     *
     * 在没有设置 inverse=true 的情况下,父子两边都维护父子关系
     *
     * 在 1-n 关系中,将 n 方设为主控方将有助于性能改善!!
     */
    @Test
    public void testOneToManyBothSave(){
        Customer customer1 = new Customer();
        customer1.setCustomerName("AA");
        Order order1 = new Order();
        order1.setOrderName("ORDER-1");
        Order order2 = new Order();
        order2.setOrderName("ORDER-2");
        //设定关联关系
        order1.setCustomer(customer1);
        order2.setCustomer(customer1);
        customer1.getOrders().add(order1);
        customer1.getOrders().add(order2);

        /* 在没有设置inverse的情况下:先插入 Customer, 再插入 Order, 3 条 INSERT, 2 条update
           因为1的那端和n的那端都维护关联关系,所有会多出update
           如果在一的那端设置inverse=true,让一的那端放弃维护关系,则只有3条insert!!
           建议设定set的invest=true,建议先插入一的那端,再插入多的那端
    */
	session.save(customer1);
	session.save(order1);
	session.save(order2);

        // 在没有设置inverse的情况下:先插入 Order, 再插入 Customer. 3 条 INSERT, 4 条 UPDATE
/*        sessionPojo.save(order1);
        sessionPojo.save(order2);
        sessionPojo.save(customer1);*/
    }
get,update,delete与上面类似
    @Test
    public void testUpdate(){
        Customer customer = (Customer) session.get(Customer.class,1);
        customer.getOrders().iterator().next().setOrderName("BBB");
    }
    @Test
    public void testCascade() {
        Customer customer = (Customer) session.get(Customer.class, 8);
        customer.getOrders().clear();
    }




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值