• Mybatis----高级映射和延迟加载


     1.订单商品数据模型

    1.1数据模型分析思路

    1、每张表记录的数据内容

      分模块对每张表记录的内容进行熟悉,相当于你学习系统 需求(功能)的过程。

    2、每张表重要的字段设置

      非空字段、外键字段

    3、数据库级别表与表之间的关系

      外键关系

    4、表与表之间的业务关系

      在分析表与表之间的业务关系时一定要建立  在某个业务意义基础上去分析。

    1.2 数据模型分析

    用户表user:

             记录了购买商品的用户信息

    订单表:orders

             记录了用户所创建的订单(购买商品的订单)

    订单明细表:orderdetail:

             记录了订单的详细信息即购买商品的信息

    商品表:items

             记录了商品信息

    表与表之间的业务关系:

             在分析表与表之间的业务关系时需要建立 在某个业务意义基础上去分析。

     先分析数据库级别之间有关系的表之间的业务关系。

    user和order:

    user---->orders:一个用户可以创建多个订单,一对多。

    orders---->user:一个订单只由一个用户创建,一对一。

     orders和orderdetail:

    orders---->orderdetail:一个订单可以包括多个订单明细,因为一个订单可以购买多个商品,每个商品的购买信息在orderdetail记录,一对多。

    orderdetail---->orders:一个订单明细只能包括在一个订单中,一对一。

    orderdetail和items:

    orderdetail---->items:一个订单明细只对应一个商品信息,一对一。

    items---->orderdetail:一个商品可以包括在多个订单明细,一对多。

    再分析数据库级别没有关系的表之间是否有业务关系:

    orders和items:

    orders和items之间可以通过orderdetail表建立 关系。

    2.一对一查询

    2.1 需求

    查询订单信息,关联查询创建订单的用户信息。

    2.2 resultType

    2.2.1 sql语句

    确定查询的主表:订单表

    确定查询的关联表:用户表

      关联查询使用内连接?还是外连接?

      由于orders表中有一个外键(user_id),通过外键关联查询用户表只能查询出一条记录,可以使用内连接。

    SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address
    FROM orders,USER WHERE USER.id=orders.user_id

    2.2.2 创建pojo

    将上边sql查询的结果映射到POJO中,POJO中必须包括所有的查询列名。

    原始的Orders.java不能映射全部字段,需要创建新的POJO。

    创建一个POJO继承包括查询字段较多的po类。

    package com.xjs.mybatis.po;
    
    /**
     * 订单的扩展类
     * @author hp
     *
     */
    //通过此类映射订单和用户查询的结果,让此类继承包括字段较多的pojo类
    public class OrdersCustom extends Orders{
        //添加用户属性
        /*USER.username,
        USER.sex,
        USER.address*/
        private String username;
        private String sex;
        private String address;
        
        public String getUsername() {
            return username;
        }
        public void setUsername(String username) {
            this.username = username;
        }
        public String getSex() {
            return sex;
        }
        public void setSex(String sex) {
            this.sex = sex;
        }
        public String getAddress() {
            return address;
        }
        public void setAddress(String address) {
            this.address = address;
        }
    
    }

    2.2.3 mapper.xml

        <!-- 查询订单关联查询用户信息 -->
        <select id="findOrdersUser" resultType="com.xjs.mybatis.po.OrdersCustom">
            SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address
                    FROM orders,USER WHERE USER.id=orders.user_id
        </select>

    2.2.4 mapper.java

        //查询订单关联查询用户信息
        public List<OrdersCustom> findOrdersUser() throws Exception;

    测试:

        @Test
        public void testFindOrdersUser() throws Exception {
            SqlSession sqlSession = sqlSessionFactory.openSession();
            //创建代理对象
            OrdersMapperCustom ordersMapperCustom=sqlSession.getMapper(OrdersMapperCustom.class);
            
            //调用mapper的方法
            List<OrdersCustom> list = ordersMapperCustom.findOrdersUser();
            System.out.println(list);
            
            sqlSession.close();
        }

    2.3 resultMap

    2.3.1 sql语句

    同resultType实现的sql

    2.3.2 使用resultMap映射的思路

    使用resultMap将查询结果中的订单信息映射到Orders对象中,在orders类中添加User属性,将关联查询出来的用户信息映射到orders对象中的user属性中。

    2.3.3 需要在Orders类中添加user属性

    2.3.4 mapper.xml

        <!-- 订单查询关联用户的resultMap 
            将整个查询结果映射到com.xjs.mybatis.po.Orders中
        -->
        <resultMap type="com.xjs.mybatis.po.Orders" id="OrdersUserResultMap">
            <!-- 配置映射的订单信息 -->
            <!-- id:如果有多个列组成唯一标识,配置多个id -->
            <id column="id" property="id"/>
            <result column="user_id" property="userId"/>
            <result column="number" property="number"/>
            <result column="createtime" property="createtime"/>
            <result column="note" property="note"/>
            
            <!-- 配置映射的关联的用户信息 -->
            <!-- association:用于映射关联查询单个对象的信息 
                property:要将关联查询的用户信息映射到Orders类中的哪个属性
            -->
            <association property="user" javaType="com.xjs.mybatis.po.User">
                <!-- id:关联查询用户的唯一标识 
                    column:指定唯一标识用户信息的列
                    property:映射到user的哪个属性
                -->
                <id column="user_id" property="id"/>
                <result column="username" property="username"/>
                <result column="sex" property="sex"/>
                <result column="address" property="address"/>
            </association>
        </resultMap>
        <!-- 查询订单关联查询用户信息,使用resultMap -->
        <select id="findOrdersUserResultMap" resultMap="OrdersUserResultMap">
            SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address
                    FROM orders,USER WHERE USER.id=orders.user_id
        </select>

    2.3.5 mapper.java

        //查询订单关联查询用户使用resultMap
        public List<Orders> findOrdersUserResultMap() throws Exception;

    测试:

        @Test
        public void testFindOrdersUserResultMap() throws Exception {
            SqlSession sqlSession = sqlSessionFactory.openSession();
            //创建代理对象
            OrdersMapperCustom ordersMapperCustom=sqlSession.getMapper(OrdersMapperCustom.class);
            System.out.println("===========");
            //调用mapper的方法
            List<Orders> list = ordersMapperCustom.findOrdersUserResultMap();
            System.out.println(list);
            
            sqlSession.close();
        }

    2.4 resultType和resultMap实现一对一查询小结

     实现一对一查询:

    resultType:使用resultType实现较为简单,如果POJO中没有包括查询出来的列名,需要增加列名对应的属性,即可完成映射。

     resultMap:需要单独定义resultMap,实现有点麻烦,如果有对查询结果有特殊的要求,使用resultMap可以完成将关联查询映射POJO属性中。

    resultMap可以实现延时加载,resultType无法实现延时加载。

    3. 一对多查询

     3.1 需求

    查询订单及订单明细的信息

     3.2 SQL语句

    确定主查询表:订单表

    确定关联查询表:订单明细表

            SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address,
                    orderdetail.id orderdetail_id,
                    orderdetail.items_id,
                    orderdetail.items_num,
                    orderdetail.orders_id
            FROM orders,
                        USER,
                        orderdetail
            WHERE USER.id=orders.user_id AND orderdetail.orders_id=orders.id

     3.3 分析

    使用resultType将上边的查询结果映射到POJO中,订单信息的就是重复。

    要求:

    对orders映射不能出现重复记录。

    在orders.java类中添加List<OrderDetail> orderDetails属性。

    最终会将订单信息映射到orders中,订单所对应的订单明细映射到orders中的ordersDetail属性中。

    3.4 在orders中添加List订单明细属性

    3.5 mapper.xml

        <!-- 查询订单关联查询用户及订单明细,使用resultMap -->
        <select id="findOrdersAndOrderDetailResultMap" resultMap="OrdersAndOrderDetailResultMap">
            SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address,
                    orderdetail.items_id,
                    orderdetail.items_num,
                    orderdetail.orders_id
            FROM orders,
                        USER,
                        orderdetail
            WHERE USER.id=orders.user_id AND orderdetail.orders_id=orders.id
        </select>

    3.6 resultMap的定义

        <!-- 查询订单及订单明细的resultMap 
            extends:使用继承,就不用在配置订单信息和用户信息的映射
         -->
        <resultMap type="com.xjs.mybatis.po.Orders" id="OrdersAndOrderDetailResultMap" extends="OrdersUserResultMap">
            <!-- 订单信息 -->
            <!-- 用户信息 -->
            
            <!-- 订单明细信息 
                一个订单关联查询出了多条明细,要使用collection进行映射
                collection:对关联查询到多条记录映射到集合对象中
                property:将关联查询到多条记录映射到com.xjs.mybatis.po.Orders类的哪个属性中
                ofType:指定映射到list集合属性中POJO的类型
            -->
            <collection property="orderdetails" ofType="com.xjs.mybatis.po.Orderdetail">
                <id column="orderdetail_id" property="id"/>
                <result column="items_id" property="itemsId"/>
                <result column="items_num" property="itemsNum"/>
                <result column="orders_id" property="ordersId"/>
            </collection>
        </resultMap>

    3.7 mapper.java

        //查询订单、用户及订单详情信息
        public List<Orders> findOrdersAndOrderDetailResultMap()throws Exception;

    3.8 小结

    mybatis使用 resultMap的 collection对关联查询的多条记录映射到一个List集合属性中。

    4. 多对多查询

     4.1 需求

    查询用户及用户购买商品的信息。

    4.2 sql语句

    查询主表是:用户表

    关联表:由于用户和商品没有直接关联,通过订单和订单明细进行关联,所以关联表:orders、orderdetail、items。

    SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address,
                    orderdetail.id orderdetail_id,
                    orderdetail.items_id,
                    orderdetail.items_num,
                    orderdetail.orders_id,
                    items.id item_id,
                    items.`name` items_name,
                    items.detail items_detail,
                    items.price items_price
    FROM orders,
                USER,
                orderdetail,
                items
    WHERE USER.id=orders.user_id AND orderdetail.orders_id=orders.id AND orderdetail.items_id=items.id

    4.3 映射思路

    将用户信息映射到user中。

    在user类中添加订单列表的属性 List<Orders> ordersList,将用户创建的订单映射到ordersList。

    在Orders中添加订单明细列表属性 LIst<OrderDetail> orderdetails,将订单的明细映射到orderdetails。

    在OrderDetail中添加items属性,将订单明细所对应的商品映射到items。

    4.4 mapper.xml

        <!-- 查询用户及购买的商品信息,使用resultMap -->
        <select id="findUserAndItemsResultMap" resultMap="UserAndItemsResultMap">
            SELECT orders.* ,
                    USER.username,
                    USER.sex,
                    USER.address,
                    orderdetail.id orderdetail_id,
                    orderdetail.items_id,
                    orderdetail.items_num,
                    orderdetail.orders_id,
                    items.`name` items_name,
                    items.detail items_detail,
                    items.price items_price
                FROM orders,
                            USER,
                            orderdetail,
                            items
                WHERE USER.id=orders.user_id AND orderdetail.orders_id=orders.id AND orderdetail.items_id=items.id
        </select>

    4.5 resultMap定义

        <!-- 查询用户及购买的商品 -->
        <resultMap type="com.xjs.mybatis.po.User" id="UserAndItemsResultMap">
            <!-- 用户信息 -->
            <id column="user_id" property="id"/>
            <result column="username" property="username"/>
            <result column="sex" property="sex"/>
            <result column="address" property="address"/>
            <!-- 订单信息
                一个用户对应多个订单,使用collection映射
             -->
             <collection property="odersList" ofType="com.xjs.mybatis.po.Orders">
                 <id column="id" property="id"/>
                 <result column="user_id" property="userId"/>
                <result column="number" property="number"/>
                <result column="createtime" property="createtime"/>
                <result column="note" property="note"/>
                 <!-- 订单明细
                     一个订单包括多个明细
                  -->
                  <collection property="orderdetails" ofType="com.xjs.mybatis.po.Orderdetail">
                      <id column="orderdetail_id" property="id"/>
                      <result column="items_id" property="itemsId"/>
                    <result column="items_num" property="itemsNum"/>
                    <result column="orders_id" property="ordersId"/>
                    <!-- 商品信息
                        一个订单明细对应一个商品 
                    -->
                    <association property="items" javaType="com.xjs.mybatis.po.Items">
                        <id column="items_id" property="id"/>
                        <result column="items_name" property="name"/>
                        <result column="items_detail" property="detail"/>
                        <result column="items_price" property="price"/>
                    </association>
                  </collection>
             </collection>
        </resultMap>

    4.6 mapper.java

        //查询用户购买商品的信息
        public List<User> findUserAndItemsResultMap()throws Exception;

    4.7 多对多查询总结

    使用resultMap是针对那些对查询结果映射有特殊要求的功能。

    5. resultMap总结

     resultType:

    作用:

      将查询结果按照sql列名POJO属性名一致映射到POJO中。

    场合:

      常见一些明细记录的展示,比如用户购买商品的明细,将关联查询信息全部展示在页面时,此时可以直接使用resultType 将每一条记录映射到POJO中,在前端页面遍历list(list中是POJO)即可。

    resultMap:

      使用association和collection完成一对一和一对多的高级映射。

    association:

    作用:

      将关联查询信息映射到一个POJO对象中。

    场合:

      为了方便查询关联信息可以使用association将关联订单信息映射为用户对象的POJO属性中,比如:查询订单及关联用户信息。

      使用resultType无法将查询结果映射到POJO对象的POJO属性中,根据对结果集查询遍历的需要选择使用resultType还是resultMap。

    collection:

    作用:

      将关联查询信息映射到一个list集合中。

    场合:

      为了方便查询遍历关联信息可以使用collection 将关联信息映射到list集合中,比如:查询用户权限范围模块下的菜单,可使用collection将模块映射到模块list中,将菜单列表映射到模块对象的菜单list属性中,这样做的目的是方便对查询结果集进行遍历查询。

      如果使用resultType无法将查询结果映射到list集合中。

    6. 延迟加载

    6.1 什么是延时加载

      resultMap可以实现高级映射(使用association、collection实现一对一及一对多映射),association、collection具备延时加载功能

    需求:

      如果查询订单并且关联查询用户信息。如果先查询订单信息即可满足要求,当我们需要查询用户信息时再查询用户信息。把对用户信息的按需去查询就是延迟加载。

    延迟加载:先从单表查询、需要时再从关联表去查询,大大提高 数据库性能,因为查询单表要比关联查询多张表速度要快。

    6.2 使用association实现延迟加载

    6.2.1 需求

    查询订单并且关联查询用户信息。

    6.2.2 mapper.xml

    需要定义两个mapper的方法对应的statement。

    1、只查询订单信息

    SELECT * FROM orders

    在查询订单的statement中使用association去延迟加载(执行)下边的statement(关联查询用户信息)

        <!-- 查询订单关联查询用户,用户信息需要延迟加载 -->
        <select id="findOrdersUserLazyLoading" resultMap="OrdersUserLazyLoading">
            SELECT * FROM orders
        </select>

    2、关联查询用户信息

      通过上边查询到的订单信息中 user_id去关联查询用户信息

    使用UserMapper.xml中的findUserById

    上边先去执行findOrdersUserLazyLoading,当需要去查询用户时再去执行findUserById,通过resultMap的定义,将延迟加载执行配置起来。

    6.2.3 延迟加载resultMap

    使用association中的select指定延迟加载去执行statement的id。

        <!-- 延迟加载的resultMap -->
        <resultMap type="com.xjs.mybatis.po.Orders" id="OrdersUserLazyLoading">
            <!-- 对订单信息进行映射配置 -->
            <id column="id" property="id"/>
            <result column="user_id" property="userId"/>
            <result column="number" property="number"/>
            <result column="createtime" property="createtime"/>
            <result column="note" property="note"/>
            
            <!-- 实现对用户信息进行延迟加载 
                select:指定延迟加载需要执行的statement的id(就是根据user_id查询用户信息的statement)
                要使用userMapper.xml中的findUserById完成用户信息的查询
                column:订单信息中关联用户信息查询的列,是user_id
                关联查询的sql理解为:
                SELECT orders.*,
                    (SELECT username FROM USER WHERE orders.user_id=USER.id) username ,
                    (SELECT sex FROM USER WHERE orders.user_id=USER.id) sex 
                FROM orders
            -->
            <association property="user" javaType="com.xjs.mybatis.po.User" select="com.xjs.mybatis.mapper.UserMapper.findUserById" column="user_id">
                <!-- 实现对用户信息进行延迟加载 -->
                
            </association>
        </resultMap>

    6.2.4 mapper.java

        //查询订单关联查询用户,用户信息延迟加载
        public List<Orders> findOrdersUserLazyLoading()throws Exception;

    6.2.5.1 测试思路

    1、执行上边mapper方法(findOrdersUserLazyLoading),内部去调用cn.itcast.mybatis.mapper.OrdersMapperCustom中的findOrdersUserLazyLoading只查询orders信息(单表)。

    2、在程序中去遍历上一步骤查询出的List<Orders>,当我们调用Orders中的getUser方法时,开始进行延迟加载。

    3、延迟加载,去调用UserMapper.xml中findUserbyId这个方法获取用户信息。

    6.2.5.2 延迟加载配置

    mybatis默认没有开启延迟加载,需要在SqlMapConfig.xml中setting配置。

        <!-- 全局配置参数,需要时在配置 -->
        <settings>
            <!-- 打开延迟加载的开关 -->
            <setting name="lazyLoadingEnabled" value="true"/>
            <!-- 将积极加载改为消极加载 ,即按需加载 -->
            <setting name="aggressiveLazyLoading" value="false"/>
        </settings>

    6.2.5.3 测试代码

        //查询订单关联查询用户,用户信息使用延迟加载
        @Test
        public void testfindOrdersUserLazyLoading() throws Exception {
            SqlSession sqlSession = sqlSessionFactory.openSession();
            OrdersMapperCustom ordersMapperCustom=sqlSession.getMapper(OrdersMapperCustom.class);
            //查询单表
            List<Orders> list = ordersMapperCustom.findOrdersUserLazyLoading();
            System.out.println(list);
            //遍历上边的订单列表
            for (Orders orders : list) {
                //实现按需加载
                User user=orders.getUser();
                System.out.println(user);
            }
            sqlSession.close();
        }

    6.2.6 延迟加载思考

    不使用mybatis提供的association及collection中的延迟加载功能,如何实现延迟加载??

    实现方法如下:

    定义两个mapper方法:

    1、查询订单列表

    2、根据用户id查询用户信息

    实现思路:

    先去查询第一个mapper方法,获取订单信息列表

    在程序中(service),按需去调用第二个mapper方法去查询用户信息。

    总之:

    使用延迟加载方法,先去查询简单的sql最好单表,也可以关联查询),再去按需要加载关联查询的其它信息。

  • 相关阅读:
    跨平台开发得力助手
    一分钟将你的WPF应用程序变身成炫彩动态Metro风格
    因.Net Target Framework导致的编译错误
    WCF笔记
    WCF元数据发布的2种方式:httpGetEnabled与mex
    使用ClickOnce发布应用程序
    TFS笔记
    升级镁光M4固态硬盘的固件
    Android调用RESTful WCF服务
    用原生的android SDK与PhoneGap分别做了个示例,发现PhoneGap的要慢得多
  • 原文地址:https://www.cnblogs.com/xjs1874704478/p/11246278.html
Copyright © 2020-2023  润新知