mybatis collection_Mybatis【配置文件】

映射文件

配置文件和映射文件还有挺多的属性我还没有讲的,现在就把它们一一补全

在mapper.xml文件中配置很多的sql语句,执行每个sql语句时,封装为MappedStatement对象,mapper.xml以statement为单位管理sql语句

Statement的实际位置就等于namespace+StatementId

占位符

在Mybatis中,有两种占位符

  • #{}解析传递进来的参数数据

  • ${}对传递进来的参数原样拼接在SQL中

主键生成策略

如果我们在Hibernate中,当我们插入数据的时候,我们是可以选择是UUID策略的…

那么在Mybatis是怎么做的呢??

UUID

    
   <insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
       <selectKey keyProperty="id" order="BEFORE" resultType="string">
           select uuid()
       selectKey>

       INSERT INTO USER(id,username,birthday,sex,address) VALUES(#{id},#{username},#{birthday},#{sex},#{address})
   insert>

主键返回

如果我们一般插入数据的话,如果我们想要知道刚刚插入的数据的主键是多少,我们可以通过以下的方式来获取

需求:

  • user对象插入到数据库后,新记录的主键要通过user对象返回,通过user获取主键值。

解决思路:

  • 通过LAST_INSERT_ID()获取刚插入记录的自增主键值,在insert语句执行后,执行select LAST_INSERT_ID()就可以获取自增主键。

mysql:

    <insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
       <selectKey keyProperty="id" order="AFTER" resultType="int">
           select LAST_INSERT_ID()
       selectKey>
       INSERT INTO USER(username,birthday,sex,address) VALUES(#{username},#{birthday},#{sex},#{address})
   insert>

oracle:

实现思路:

  • 先查询序列得到主键,将主键设置到user对象中,将user对象插入数据库。

    
   <insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
       <selectKey keyProperty="id" order="BEFORE" resultType="int">
           select 序列.nextval() from dual
       selectKey>

       INSERT INTO USER(id,username,birthday,sex,address) VALUES( 序列.nextval(),#{username},#{birthday},#{sex},#{address})
   insert>

resultMap

有的时候,我们看别的映射文件,可能看不到以下这么一段代码:

    <resultMap id="userListResultMap" type="user" >
       
         <id column="id_" property="id"/>
         
         <result column="username_" property="username"/>
         <result column="birthday_" property="birthday"/>

    resultMap>

因为,如果我们的数据表的字段和JavaBean的属性名称是相同时,我们就不用上面那段代码了。Mybatis会自动帮我们把返回的结果进行封装成JavaBean

那当我们数据表的字段和JavaBean的属性名称不是相同时,我们就需要使用resultMap,也就是上面那段代码

  • 当然了,在正常情况下列名和JavaBean的属性名一般都是不同的,因此还是需要resultMap的。

resultMap和resultType区别

resultType :指定输出结果的类型(pojo、简单类型、hashmap..),将sql查询结果映射为java对象 。

  • 使用resultType注意:sql查询的列名要和resultType指定pojo的属性名相同,指定相同 属性方可映射成功,如果sql查询的列名要和resultType指定pojo的属性名全部不相同,list中无法创建pojo对象的。

resultMap:将sql查询结果映射为java对象。

  • 如果sql查询列名和最终要映射的pojo的属性名不一致,使用resultMap将列名和pojo的属性名做一个对应关系 (列名和属性名映射配置)

使用resultMap

    <resultMap id="userListResultMap" type="user" >
       
         <id column="id_" property="id"/>
         
         <result column="username_" property="username"/>
         <result column="birthday_" property="birthday"/>

    resultMap>
9870611662247c6b24b4878f925356f8.png
这里写图片描述
35cf5b6f601735d79fd073f055b6f4d2.png
这里写图片描述

resultType和resultMap用法总结

resultType:

  • 作用:

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

  • 场合:

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

resultMap:

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


association:

  • 作用:

    • 将关联查询信息映射到一个pojo类中。

  • 场合:

    • 为了方便获取关联信息可以使用association将关联订单映射为pojo,比如:查询订单及关联用户信息。

collection:

  • 作用:

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

  • 场合:

    • 为了方便获取关联信息可以使用collection将关联信息映射到list集合中,比如:查询用户权限范围模块和功能,可使用collection将模块和功能列表映射到list中。


Collection在前面好像并没有用过,下面就看一下它的用法:

Order与OrderDetails关系

package cn.itcast.mybatis.po;

import java.io.Serializable;
import java.util.Date;
import java.util.List;

public class Orders implements Serializable {
   private Integer id;

   private Integer userId;

   private String number;

   private Date createtime;

   private String note;

   //关联用户信息
   private User user;

   //订单明细
   private List orderdetails;public Integer getId() {return id;
   }public void setId(Integer id) {this.id = id;
   }public Integer getUserId() {return userId;
   }public void setUserId(Integer userId) {this.userId = userId;
   }public String getNumber() {return number;
   }public void setNumber(String number) {this.number = number == null ? null : number.trim();
   }public Date getCreatetime() {return createtime;
   }public void setCreatetime(Date createtime) {this.createtime = createtime;
   }public String getNote() {return note;
   }public void setNote(String note) {this.note = note == null ? null : note.trim();
   }public User getUser() {return user;
   }public void setUser(User user) {this.user = user;
   }public ListgetOrderdetails() {return orderdetails;
   }public void setOrderdetails(List orderdetails) {this.orderdetails = orderdetails;
   }
}

SQL语句

     
    <select id="findOrderAndOrderDetails" resultMap="orderAndOrderDetails" >
           SELECT
     orders.*,
     user.username,
     user.sex ,
     orderdetail.id orderdetail_id,
     orderdetail.items_num,
     orderdetail.items_id
   FROM
     orders,
     USER,
     orderdetail
   WHERE orders.user_id = user.id  AND orders.id = orderdetail.orders_id
    select>

resultMap

    
   <resultMap type="orders" id="orderAndOrderDetails" extends="ordersUserResultMap">
       

       
       <collection property="orderdetails" ofType="cn.itcast.mybatis.po.Orderdetail">
           
           <id column="orderdetail_id" property="id"/>
           <result column="items_num" property="itemsNum"/>
           <result column="items_id" property="itemsId"/>
       collection>

   resultMap>

一般地使用resultMap会多一点。


Mybatis映射文件处理特殊字符

第一种方法:

  • 用了转义字符把>和

  • SELECT * FROM test WHERE 1 = 1 AND start_date  <= CURRENT_DATE AND end_date >= CURRENT_DATE

第二种方法:

配置文件

别名

typeAliases别名:

80373076e2c81cfc08f3332484051436.png
这里写图片描述

自定义别名:

    
   <typeAliases>
       
       
       
       <package name="cn.itcast.mybatis.po"/>

   typeAliases>

Mapper加载

    <mappers>
       
       <mapper resource="sqlmap/User.xml" />
       
       
       
       
       <package name="cn.itcast.mybatis.mapper"/>
   mappers>

延迟加载

在进行数据查询时,为了提高数据库查询性能,尽量使用单表查询,因为单表查询比多表关联查询速度要快。

如果查询单表就可以满足需求,一开始先查询单表,当需要关联信息时,再关联查询,当需要关联信息再查询这个叫延迟加载

在Mybatis中延迟加载就是在resultMap中配置具体的延迟加载..

c853d6d1454b3d501e45bb8041023cd2.png
这里写图片描述

在Mybatis的文件中配置全局延迟加载

    
   <settings>
       
       <setting name="lazyLoadingEnabled" value="true" />  
       
       <setting name="aggressiveLazyLoading" value="false" />
   settings>

延迟加载测试

当需要用户时调用 Orders类中的getUser()方法执行延迟加载 ,向数据库发出sql。

由于是对User进行延迟加载,那么我们只要查询Orders相关的信息即可了

     
    <select id="findOrderUserListLazyLoading" resultMap="orderCustomLazyLoading">
        SELECT
         orders.*
       FROM
         orders
    select>

使用resultMap来配置延迟加载

    
   <resultMap type="orders" id="orderCustomLazyLoading">
       
       
       <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="user"select="cn.itcast.mybatis.mapper.UserMapper.findUserById" column="user_id">association>

   resultMap>
29740be9bb4317cd09a368e3a5d6c760.png
这里写图片描述

总结

  • 在程序中调用的SQL语句是由映射文件的命令空间+sql片段的id所组成的。它内部会生成一个Statement对象的。

  • 在使用别名的时候,可以指定包名,在使用总配置文件加载映射文件时,也可以指定包名。

  • 主键如果需要返回的话,使用selectKey 标签即可。UUID也可以返回。在Oracle的话,是使用序列来返回自动增长的主键的。

  • 占位符有两种,一种是解析传递进来的参数数据、一种是原样输出传递进来的数据。

如果文章有错的地方欢迎指正,大家互相交流。习惯在微信看技术文章,想要获取更多的Java资源的同学,可以关注微信公众号:Java3y

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值