Mybatis进阶教程

简介: 前言接着上一篇Mybatis入门继续,上一篇主要演示了Mybatis的基本操作,对数据库的增删改查,但是在实际项目中用到的Mybatis知识点要远多于这些基本操作,这篇将演示一些Mybatis的更加常用的用法。

前言

接着上一篇Mybatis入门继续,上一篇主要演示了Mybatis的基本操作,对数据库的增删改查,但是在实际项目中用到的Mybatis知识点要远多于这些基本操作,这篇将演示一些Mybatis的更加常用的用法。

知识点汇总

新建项目,基本环境都如上篇,Maven的pom.xml中添加jUnit依赖。

Insert获取主键的值

Mapper.xml的Insert节点添加useGeneratedKeys,keyProperty即可。

KeyProperty:(仅对 insert 和 update 有用)唯一标记一个属性,MyBatis 会通过 getGeneratedKeys 的返回值或者通过 insert 语句的 selectKey 子元素设置它的键值,默认:unset。如果希望得到多个生成的列,也可以是逗号分隔的属性名称列表。

UseGeneratedKeys:仅对 insert 和 update 有用)这会令 MyBatis 使用 JDBC 的 getGeneratedKeys 方法来取出由数据库内部生成的主键(比如:像 MySQL 和 SQL Server 这样的关系数据库管理系统的自动递增字段),默认值:false。

Mybaitis 入门教程的项目, Mapper.xml文件中
修改内容为下:

    <insert id="insertBlog" parameterType="Blog"
            keyProperty="id"
            useGeneratedKeys="true"
    >
        INSERT INTO Blog (id,title) VALUES (#{id},#{title})
    </insert>

此时数据库内容:

img_9b64f329a77866add20099a8197c4b5e.png
程序运行前数据库内容

测试程序:

 @Test
    public void test1(){
        InputStream inputStream = null;
        SqlSession sqlSession = null;
        try {
            inputStream = Resources.getResourceAsStream("mybatis-config.xml");
            SqlSessionFactory mSqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
            sqlSession = mSqlSessionFactory.openSession();
            Blog blog = new Blog("获取主键的值");
            sqlSession.insert("me.aihe.dao.BlogMapper.insertBlog",blog);
            // 检测是否将插入数据的主键返回
            System.out.println(blog.getId());
            sqlSession.commit();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (sqlSession != null) {
                sqlSession.close();
            }
        }
    }

运行结果

img_512bdf28ff7f5785f98f8dc571fe26aa.png
运行结果
img_3e3bcfe7004a2369b25390d8a8b437a4.png
运行程序之后数据库内容

注意:还有一种实现方式是在insert内部添加<SelectKey>节点,这里就不再演示。

SQL映射文件参数分析

在Mapper.xml文件中,SQL语句里我们可以传入对象,或传入其它参数。
当我们以如下方式传递参数时:

  // 这是在UserMapp.java文件
  int insertByParam(int id,String name, String email);
 //UserMapper.xml文件中
    <insert id="insertByParam">
        INSERT INTO User (id,name,email) VALUE (#{id},#{name},#{email})
    </insert>

测试程序如下:

@Test
    public void test2(){
        InputStream inputStream = null;
        SqlSession sqlSession = null;
        try {
            inputStream = Resources.getResourceAsStream("mybatis-config.xml");
            SqlSessionFactory mSqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
            sqlSession = mSqlSessionFactory.openSession();

            UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
            userMapper.insertByParam(2,"Gao","gao@gmail.com");

            sqlSession.commit();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (sqlSession != null) {
                sqlSession.close();
            }
        }
    }

这个时候程序运行会出现如下错误,提示我们找不到参数。

### Cause: org.apache.ibatis.binding.BindingException: Parameter 'id' not found. Available parameters are [arg2, arg1, arg0, param3, param1, param2]
解决方法:四种任选其一
  • 修改UserMapp.java文件添加@Param注解,即可
int insertByParam(@Param("id") int id,@Param("name") String name, @Param("email") String email);
  • 修改UserMapper.xml文件,修改为param*类型如下
    <insert id="insertByParam"  >
        INSERT INTO User (id,name,email) VALUE (#{param1},#{param2}, #{param3})
    </insert>
  • 将参数换为对象,如下
// UserMapper.java修改为如下
    int insertByPoDo(User user);

//UserMapper.java修改为下
   <insert id="insertByPoDo"
        parameterType="User"
    >
        INSERT INTO User (name,email) VALUE (#{name},#{email})
    </insert>

//Test文件修改为如下
    UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
    User user = new User("leishen","Rxlen@126.com");
    userMapper.insertByPoDo(user);
  • 将参数转为Map对象。
     int insertByMap(Map<String,Object> map);

//xml
      <insert id="insertByMap">
          INSERT INTO User (name,email) VALUE (#{name},#{email})
      </insert>

//Test
            UserMapper userMapper = sqlSession.getMapper(UserMapper.class);

            HashMap<String,Object> map = new HashMap<String, Object>();
            map.put("name","test");
            map.put("email","test@qq.com");
            userMapper.insertByMap(map);

            sqlSession.commit();

一般来说,模型与业务相关传入POJO对象,与业务无关就传入Map对象。
注意:关于参数封装原理部分请查看MapperMethod.java,ParamNameResolve.java

SQL语句中$与#的区别

默认情况下,使用#{}格式的语法会导致 MyBatis 创建预处理语句属性并安全地设置值(比如?)。这样做更安全,更迅速,通常也是首选做法,不过有时你只是想直接在 SQL 语句中插入一个不改变的字符串。比如,像 ORDER BY,你可以这样来使用:

ORDER BY ${columnName}

这里 MyBatis 不会修改或转义字符串。
注意:** 以这种方式接受从用户输出的内容并提供给语句中不变的字符串是不安全的,会导致潜在的 SQL 注入攻击,因此要么不允许用户输入这些字段,要么自行转义并检验。**

$是字符串替换,#会进行预编译

获取列表数据

// UserMapper接口
    List<User> getUserByEmail(@Param("email") String email);

// UserMaper.xml
    <select id="getUserByEmail" resultType="me.aihe.dao.User">
        SELECT * FROM User WHERE email LIKE #{email}
    </select>

//测试文件
   UserMapper userMapper = sqlSession.getMapper(UserMapper.class);

   List<User> result = userMapper.getUserByEmail("aihe%");
   System.out.println(result);

最终输出内容

img_6a822255c22796e5954839f88c603804.png
获取列表数据结果

获取Map数据

//UserMapeer接口文件
    Map<String,Object> getUserMapById(@Param("id") Integer id);

//UserMapper.xml文件
    <select id="getUserMapById" resultType="java.util.Map">
        SELECT * FROM User WHERE id = #{id}
    </select>

//测试程序
    UserMapper userMapper = sqlSession.getMapper(UserMapper.class);

    Map result = userMapper.getUserMapById(2);
    System.out.println(result);

测试输出结果

img_c3b3f9c4a012b636b6e7b0560275fbce.png
获取Map数据结果

获取Map映射的多组数据

这个和获取列表很类似了,只不过把数组都封装到一个Map中去了

// UserMapper接口文件
    @MapKey("id")
    Map<Integer,User> getUserMapLikeemail(@Param("email") String email);

// UserMapper.xml文件

    <select id="getUserMapById" resultType="java.util.Map">
        SELECT * FROM User WHERE id = #{id}
    </select>

// 测试文件
    UserMapper userMapper = sqlSession.getMapper(UserMapper.class);

    Map result = userMapper.getUserMapLikeemail("%aihe%");
    System.out.println(result);

程序输出

img_9e918f1a2cb313ccc8ab242c1aa05195.png
获取Map映射的多组数据

**注意: 注解@MapKey指定的是以数据库表中的那一列作为Map结果的key。

使用ResultMap来映射结果数据

// UserMapper接口
    User getUserMapByIdThroughResultMap(@Param("id") Integer id);

// UserMapper.xml文件
    //Id标识的是查询结果中的主键是那一列,对应的User属性是什么
    <resultMap id="MyResultMap" type="User">
        <id column="id" property="id" />    
        <result property="email" column="email" />
        <result property="name" column="name" />
    </resultMap>

    <select id="getUserMapByIdThroughResultMap" resultMap="MyResultMap">
        SELECT * FROM User WHERE id = #{id}
    </select>

//测试文件
   UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
   User user = userMapper.getUserMapByIdThroughResultMap(2);
   System.out.println(user);

输出结果

img_4192b1fdd82e9c51d123ab5b96c68e1f.png
ResultMap映射结果数据

总结

本文主要讲解了几个Mybatis日常使用中经常用到的几个知识点,获取主键,参数映射,及$与#符号的区别,最后演示一些各种获取数据库中结果的方式。

参考

Mybatis 官方文档
Mybatis 视频教程

目录
相关文章
|
5月前
|
XML Java 数据库连接
MyBatis--映射关系一对一和MyBatis--映射关系多对一 -都有基于xml和注解的教程
MyBatis--映射关系一对一和MyBatis--映射关系多对一 -都有基于xml和注解的教程
91 0
|
2月前
|
XML Java 数据库连接
Mybatis-Plus学习小项目及详细教程
Mybatis-Plus学习小项目及详细教程
|
3月前
|
SQL XML Java
【JavaEE进阶】 MyBatis之动态SQL
【JavaEE进阶】 MyBatis之动态SQL
|
3月前
|
XML Java 数据库连接
【JavaEE进阶】 MyBatis使用XML实现增删改查
【JavaEE进阶】 MyBatis使用XML实现增删改查
|
3月前
|
Java 关系型数据库 数据库连接
【JavaEE进阶】 MyBatis使用注解实现增删改查
【JavaEE进阶】 MyBatis使用注解实现增删改查
|
3月前
|
Java 关系型数据库 数据库连接
【JavaEE进阶】MyBatis⼊⻔
【JavaEE进阶】MyBatis⼊⻔
|
3月前
|
XML 监控 druid
【Java专题_02】springboot+mybatis+pagehelper分页插件+druid数据源详细教程
【Java专题_02】springboot+mybatis+pagehelper分页插件+druid数据源详细教程
|
3月前
|
SQL Java 数据库连接
【MyBatisPlus】通俗易懂 快速入门 详细教程
【MyBatisPlus】通俗易懂 快速入门 详细教程
83 0
|
4月前
|
Java 数据库连接 数据库
Spring Boot整合Mybatis Plus[极简教程]
Spring Boot整合Mybatis Plus[极简教程]
60 0
|
4月前
|
SQL 缓存 Java
Mybatis保姆级丝滑教程(一文搞懂系列)(2)
Mybatis保姆级丝滑教程(一文搞懂系列)
173 0