Mybatis动态sql中@Param使用详解

 更新时间:2023年10月08日 08:32:25   作者:riun、  
这篇文章主要介绍了Mybatis动态sql中@Param使用详解,当方法的参数为非自定义pojo类型,且使用了动态sql,那么就需要在参数前加上@Param注解,需要的朋友可以参考下

Mybatis中的@param注解的使用场景

1、方法有多个参数

2、方法参数要取别名

3、XML 中的 SQL 使用了 $

4、动态sql中参数是非自定义pojo类型

前三种,相信大家都非常熟练,这里不再多说,这里主要说下第四种。

方法的参数为非自定义pojo类型,且使用了动态sql,那么就需要在参数前加上@Param注解。

测试:(此处使用SpringBoot2.0+ 、Mybatis-Plus)

数据库数据

在这里插入图片描述

代码

pojo:

/**
 * @author: HanXu
 * on 2019/12/18
 * Class description:
 */
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Test {
    private Integer id;
    private String name;
    private Integer age;
    private Integer num;
}

mapper接口:

/**
 * @author: HanXu
 * on 2019/12/18
 * Class description:
 */
public interface TestMapper extends BaseMapper<Test> {
    Test sel(Test t);
    Test seli(Integer id);
    Test seli2(@Param("id")Integer id);
    Test sels(String s);
    Test sels2(@Param("s")String s);
}

对应的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="hx.insist.demo.mapper.TestMapper">
    <select id="sel" parameterType="hx.insist.demo.domain.Test" resultType="hx.insist.demo.domain.Test">
        select * from test
        where 1 = 1
        <if test="id!=null">
            and id = #{id}
        </if>
        <if test="name!=null">
            and name = #{name}
        </if>
        <if test="age!=null">
            and age = #{age}
        </if>
        <if test="num!=null">
            and num = #{num}
        </if>
    </select>
    <select id="seli" parameterType="Integer" resultType="hx.insist.demo.domain.Test">
        select * from test
        where 1 = 1
        <if test="id!=null">
            and id = #{id}
        </if>
    </select>
    <select id="seli2" parameterType="Integer" resultType="hx.insist.demo.domain.Test">
        select * from test
        where 1 = 1
        <if test="id!=null">
            and id = #{id}
        </if>
    </select>
    <select id="sels" resultType="hx.insist.demo.domain.Test">
        select * from test
        where 1 = 1
        <if test="s!=null">
            and name = #{s}
        </if>
    </select>
    <select id="sels2" resultType="hx.insist.demo.domain.Test">
        select * from test
        where 1 = 1
        <if test="s!=null">
            and name = #{s}
        </if>
    </select>
</mapper>

service层:

/**
 * @author: HanXu
 * on 2019/12/18
 * Class description:
 */
@Service
public class TestService {
    @Resource
    private TestMapper testMapper;
    //测试自定义的存在setter方法的对象
    public void selTest(){
         Test t = new Test();
        t.setId(1);
        Test sel = testMapper.sel(t);
        System.out.println(sel);
    }
    //测试Integer
    public void seliTest(){
        Test seli = testMapper.seli(1);
        System.out.println(seli);
    }
    //测试String
    public void selsTst(){
        Test aaa = testMapper.sels("aaa");
        System.out.println(aaa);
    }
    //测试Integer,且写了@Param
    public void seliTest2(){
        Test seli = testMapper.seli2(1);
        System.out.println(seli);
    }
    //测试String,且写了@Param
    public void selsTest2(){
        Test test = testMapper.sels2("aaa");
        System.out.println(test);
    }
}

controller层:

@GetMapping("testT")
    public ResponseEntity test(){
        testService.selTest();
        return ResponseEntity.ok().build();
    }
    @GetMapping("testI")
    public ResponseEntity testi(){
        testService.seliTest();
        return ResponseEntity.ok().build();
    }
    @GetMapping("testS")
    public ResponseEntity tests(){
        testService.selsTst();
        return ResponseEntity.ok().build();
    }
    @GetMapping("testI2")
    public ResponseEntity testI2(){
        testService.seliTest2();
        return ResponseEntity.ok().build();
    }
    @GetMapping("testS2")
    public ResponseEntity testS2(){
        testService.selsTest2();
        return ResponseEntity.ok().build();
    }

测试结果

测试Test seli(Integer id);

在这里插入图片描述

在这里插入图片描述

测试Test seli2(@Param(“id”)Integer id);

在这里插入图片描述

在这里插入图片描述

测试Test sels(String s);

在这里插入图片描述

在这里插入图片描述

测试Test sels2(@Param(“s”)String s);

在这里插入图片描述

在这里插入图片描述

测试Test sel(Test t);

在这里插入图片描述

在这里插入图片描述

结论

上述5个测试用例中 分为三对:

    Test sel(Test t);//5
    Test seli(Integer id);//1
    Test seli2(@Param("id")Integer id);//2
    Test sels(String s);//3
    Test sels2(@Param("s")String s);//4

1和2是一对测试,除了2比1多了一个@Param注解之外其他全部相同,测试结果表示:加了注解的能正确执行,未加注解的不能正确执行。

3和4是一对测试,除了4比3多了一个@Param注解之外其他全部相同,测试结果表示:加了注解的能正确执行,未加注解的不能正确执行。

5和1、3是一对测试,不同之处在于5中的参数是自定义pojo,里面每个属性都有setter、getter方法,结果表示:在不加@Param注解的情况下,自定义pojo作为参数可以正常执行,非自定义pojo(没有参数的getter方法)作为参数不能正常执行。

以上基于sql语句中使用了动态sql(存在不确定的语句)

我的猜想:

这跟Mybatis内部实现有关。当你使用动态sql时,#{}去拿你的参数时,默认将参数当成了pojo类型,去取对应属性的getter方法,而这时如果你的参数不是pojo类型,是一个java定义好的类型,比如参数为Integer id,String s,使用#{}得到它的值时,Mybatis就会去直接取相应的getter方法,比如反射Integer类,取getId()方法 、反射String类,取getS()方法,但是这些类中并没有相应的方法,那么就会报错:

org.apache.ibatis.reflection.ReflectionException: There is no getter for property named ‘id’ in ‘class java.lang.Integer’
org.apache.ibatis.reflection.ReflectionException: There is no getter for property named ‘s’ in ‘class java.lang.String’

到此这篇关于Mybatis动态sql中@Param使用详解的文章就介绍到这了,更多相关Mybatis动态sql内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java 如何快速,优雅的实现导出Excel

    Java 如何快速,优雅的实现导出Excel

    这篇文章主要介绍了Java 如何快速,优雅的实现导出Excel,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下
    2021-03-03
  • Activiti进阶之组任务实现示例详解

    Activiti进阶之组任务实现示例详解

    这篇文章主要为大家介绍了Activiti进阶之组任务实现示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 详解Java中“==”与equals()的区别

    详解Java中“==”与equals()的区别

    这篇文章主要介绍了详解Java中“==”与equals()的区别的相关资料,需要的朋友可以参考下
    2017-02-02
  • 详解Java实现JSONArray转Map的三种实现方式

    详解Java实现JSONArray转Map的三种实现方式

    本文主要介绍了Java实现JSONArray转Map的三种实现方式,本文只是自己常用的三种,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • springboot整合shardingjdbc实现分库分表最简单demo

    springboot整合shardingjdbc实现分库分表最简单demo

    我们知道分库分表是针对某些数据量持续大幅增长的表,比如用户表、订单表等,而不是一刀切将全部表都做分片,这篇文章主要介绍了springboot整合shardingjdbc实现分库分表最简单demo,需要的朋友可以参考下
    2021-06-06
  • 微服务实战之怎样提升springboot服务吞吐量

    微服务实战之怎样提升springboot服务吞吐量

    这篇文章主要介绍了微服务实战之怎样提升springboot服务吞吐量方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • Spring定时任务实现与配置(二)

    Spring定时任务实现与配置(二)

    这篇文章主要为大家详细介绍了Spring定时任务的实现与配置第二篇,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • 基于java读取并引用自定义配置文件

    基于java读取并引用自定义配置文件

    这篇文章主要介绍了基于java读取并引用自定义配置文件,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-06-06
  • SpringBoot集成Druid连接池进行SQL监控的问题解析

    SpringBoot集成Druid连接池进行SQL监控的问题解析

    这篇文章主要介绍了SpringBoot集成Druid连接池进行SQL监控的问题解析,在SpringBoot工程中引入Druid连接池非常简单,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下
    2021-07-07
  • SpringBoot的@EnableAsync和@Async注解分析

    SpringBoot的@EnableAsync和@Async注解分析

    这篇文章主要介绍了SpringBoot的@EnableAsync和@Async注解分析,Spring Boot是一个快速开发框架,可以帮助开发人员快速构建基于Spring的应用程序,需要的朋友可以参考下
    2023-07-07

最新评论