XiaodaBlogSource/source/_posts/Mybatis基本介绍.md
2025-03-19 21:19:20 +08:00

11 KiB
Raw Blame History

title, date, author
title date author
Mybatis基本介绍 2022-11-18 10:30:31 文永达

Mybatis


Maven引入Mybatis

版本号最好去Maven Repository中查找

<properties>
    <mybatis.version>3.5.7</mybatis.version>
</properties>
<dependencies>
	<dependency>
    	<groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>${mybatis.version}</version>
    </dependency>
</dependencies>

Mybatis配置文件 mybatis-config.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>

    <typeAliases>
        <!--<typeAlias type="com.crx.entity.User" alias="user"></typeAlias>-->
        <!-- 默认值就是类型首字母小写 -->
        <package name="com.wyd.mybatis20210702.entity"/>
    </typeAliases>

    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC"></transactionManager>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost:3306/ssm?useSSL=false&amp;useUnicode=true&amp;characterEncoding=utf8"/>
                <property name="username" value="root"/>
                <property name="password" value="123456"/>
            </dataSource>
        </environment>
    </environments>

    <!-- 注册mapper映射文件 -->
    <mappers>
        <mapper resource="mapper/UserMapper.xml"></mapper>
        <!-- 注册接口 -->
        <!--<mapper class="com.wyd.dao.UserMapper2"></mapper>-->
    </mappers>

</configuration>

Springboot中 application.yml

spring:
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/ssm?useSSL=false&amp;useUnicode=true&amp;characterEncoding=utf8&serverTimezone=Asia/Shanghai
    username: root
    password: 123456

mybatis:
  type-aliases-package: com.wyd.mybatis20210702.entity
  mapper-locations: classpath*:/mapper/*.xml

Mybatis interface UserMapper

package com.wyd.mybatis20210702.dao;

import com.wyd.mybatis20210702.entity.User;

public interface UserMapper {
    User selectUserById(Integer id);

    int insertUser(User user);

    int updateUser(User user);

    int deleteUser(Integer id);
}

每个Mapper接口都有对应的xml映射文件如果idea安装有MybatisX插件可以单击类名 ALT+ENTER 快捷键即可创建对应xml映射文件

映射文件夹一般命名为mapper 存放于resources文件夹下

在Mapper接口每声明一个方法可以通过MybatisX创建对应的映射但命名需遵循一定规范否则需自己选择crud

Mybatis xml映射文件 UserMapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
<mapper namespace="com.wyd.mybatis20210702.dao.UserMapper">
    <select id="selectUserById" resultType="user">
        SELECT id,username,PASSWORD,birthday FROM USER
    </select>

    <insert id="insertUser" parameterType="com.crx.entity.User">
        insert into user (username,password) values (#{username},#{password})
    </insert>

    <update id="updateUser" parameterType="com.crx.entity.User">
        update user set username = #{username},password = #{password},birthday = #{birthday} where id = #{id}
    </update>

    <delete id="deleteUser">
        delete from user where id = #{id}
    </delete>
</mapper>

可以发现xml映射文件头跟mybatis-config.xml配置文件头很像

就有一个单词不同

一个是config 一个是mapper这决定了这个xml文件可以写什么语句

每个参数都接#{}{}里就写接口中的参数名

重点来了

#{}和${}的区别

学过jdbc知道有Statement和PreparedStatement

一个是固定好的sql语句一个是预处理语句

PreparedStatement #{}就很像 ? 后面接参数名,这就很好地防止了 Sql Injection 也就是sql'注入

我这些sql语句都没有写${}就是因为防止sql注入开发时严谨使用

1、Statement和PreparedStatement区别是一样 2、#代表使用的底层是PreparedStatement。占位符的写法预加载 3、$代表使用的底层是Statement。字符串的拼接的写法。SQL注入

那么什么时候使用${}呢?最多的是模糊查询,但是也不推荐这样用

<select id="selectAllUsersByUsername" parameterType="string" resultType="com.crx.entity.User">
        select id,username,password,birthday from user where username like '%${username}%'
</select>

为了拼接上参数就必须使用${}来固化sql

应该在Java代码上来实现模糊查询

String username = ;
if(username != null && !"".equals(username.trim())){
            username = "%" + username + "%";
        }else{
            username = "%%";
}

这样就可以避免sql注入


resultType 就是 返回的数据类型有点儿Java方法中的return 不过这是返回的类型而不是变量名

parameterType 就是参数类型,实体类要写包名类名

而Java基本数据类型比如int就直接写int就行了引用数据类型比如IntegerString 要把首字目小写integerstring这是因为用到了别名也可以像实体类一样写java.lang.Integerjava.lang.String

Mybatis xml映射文件特性

动态sql

应用场景,条件查询

<select id="selectUsers" resultType="com.crx.entity.User">
        select id,username,password,birthday from user
       <!-- if标签相当于多重if只要条件全都满足所有满足的sql代码块都会执行 -->
        <where>
            <if test="id != null">
                and id = #{id}
            </if>
            <if test="username != null">
                and username = #{username}
            </if>
            <if test="password != null">
                and password = #{password}
            </if>
        </where>
    </select>

标签可动态拼接条件如果where后条件字段都为null则where以及后面语句都不会拼接

标签则相当于if判断,test等于约束条件需要注意的是其中只要写字段名即可


<select id="selectUsers" resultType="com.crx.entity.User">
        select id,username,password,birthday from user
        /*
            when,choose,otherwise相当于if...else if....else if
            otherwise相当于else
            当满足第一个条件时,不再向下执行判断
        */
        <where>
            <choose>
                <when test="username != null and username != ''">
                    and username = #{username}
                </when>
                <when test="password != null and password != ''">
                    and password = #{password}
                </when>
                <otherwise>
                    and id = #{id}
                </otherwise>
            </choose>
        </where>
    </select>

标签有点像Java中的switch语句其中的标签则向当于case但这个case中是带有break的也就是说每满足一个条件则此标签后续语句都不会执行有一点需要说的是中可以有多个满足一个则会跳出这个标签继续执行后续语句如果都不满足如果有标签则会执行这有点儿像default语句了


<select id="selectUsers" resultType="com.crx.entity.User">
        select id,username,password,birthday from user
       /*
            有时候我们需要去掉一些特殊的SQL语法比如说andorwhere
            此时可以使用trim标签
        */
        <!--<trim prefix="where" prefixOverrides="and">
            <if test="username != null and username != '' ">
                username = #{username}
            </if>
            <if test="password != null and password != ''">
                and password = #{password}
            </if>
        </trim>-->
    </select>

这个例子中,标签中的prefix属性会使where 元素会动态地在行首插入 where关键字prefixOverrides属性会删掉额外的and这些and是在使用条件语句给列赋值时引入的

如果都不满足则不会插入where关键字


Update同样适用动态sql

<update id="updateUser" parameterType="com.crx.entity.User">
        update user
        <set>
            <if test="username != null and username != '' ">
                username = #{username}
            </if>
            <if test="password != null and password != ''">
                ,password = #{password}
            </if>
            where id = #{id}
      </set>
</update>

这个例子中,set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号(这些逗号是在使用条件语句给列赋值时引入的)。

也相当于上面标签

<trim prefix="SET" suffixOverrides=",">
  <if test="username != null and username != '' ">
         username = #{username}
      </if>
      <if test="password != null and password != ''">
         ,password = #{password}
      </if>
      where id = #{id}
</trim>

动态 SQL 的另一个常见使用场景是对集合进行遍历

Mapper接口

List<User> selectUserByIds(@Param("ids") List<Integer> ids);

这里需要注意的是@Param注解必须加上Mybatis内部解析参数时会把接口中的方法形参列表当作一个Map 当参数不是一个时key是arg0,arg1,...或param1,param2,...

<select id="selectUserByIds" resultType="com.crx.entity.User">
        <include refid="selectUser"></include>
        where id in
        <foreach collection="ids" open="(" close=")" separator="," item="id">
            #{id}
        </foreach>
</select>

这里看到标签不要慌这是引入公共sql语句很像Java中的引包多用于重复的sql语句减少代码量增加简洁性

<sql id="selectUser">
        select id,username,password,birthday from user
</sql>

使用标签来定义公共sql其中id属性是这个公共sql独有的命名切忌命名不能重复

引用就看上面的集合遍历实例

再继续看集合遍历实例

先把想要生成的sql语句写上

select id,username,password,birthday from user where id in(1,2,3,5)

看到in了吧这是包含意思是查询id 是1235符合条件的行

相当于 id = 1 or id =2 or id = 3 or id = 5

这里用到了标签是不是很熟悉确实很像JavaScript中的forEach这个标签中collection属性就是需要遍历的对象名,

那么后面的openclose属性是干啥的顾名思义前者相当于前缀suffix后者就是后缀了presuffix了相当于这个标签最终生成的sql语句前后都有(),separator属性每遍历出一个元素就会在元素后加上逗号最后的元素不加,item属性就是生成sql语句参数名

也就是往#{id}中传值