MyBatis动态推理参数类型

news2024/9/21 14:49:25

前言

思考一个问题,前面的#{}和${}的区别中,我们知道了#{},MyBatis底层调用的是preparestatement这种预编译的方式,这种方式sql语句会预先编程 select * from t_user where id = ?这种形式,随后调用setInt(),setString(),…等方法为?赋值,但是myBatis是怎样知道参数是什么类型呢?这个就是MyBatis的动态推理了。具体是怎样推理的我们不再讨论,接下来我们讨论有哪些数据类型,MyBatis可以动态推理出来。

接下来通过各种查询来测试

准备工作
在这里插入图片描述
在这里插入图片描述

简单数据类型且只有一个参数

String

接口中方法

/***
     * 通过姓名来查询
     * @param id
     * @return
     */
    List<Student> selectById(Long id);

Sql映射文件中代码(完整写法,告诉MyBatis我们的参数是什么类型的)

<select id="selectByName" resultType="Student" parameterType="java.lang.String">
        select * from t_student where name = #{name}
    </select>

测试代码

@Test
    public void testSelectByName(){
        SqlSession sqlSession = SqlSessionUtil.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        List<Student> students = mapper.selectByName("张三");
        for (Student s : students){
            System.out.println(s);
        }
        sqlSession.close();
    }

测试结果
在这里插入图片描述
可以看出可以正常的查出来

缺省一下parameterType

 <select id="selectByName" resultType="Student">
        select * from t_student where name = #{name}
    </select>

再次运行结果
在这里插入图片描述
可以看出MyBatis可以正确地推导出参数类型

java.util.Date

接口中方法

/***
     * 根据birth查询
     * @param birth
     * @return
     */
    List<Student> selectByBirth(Date birth);

sql语句映射文件中代码(缺省版)

<select id="selectByBirth" resultType="Student">
        select * from t_student where birth = #{birth}
    </select>

测试代码

@Test
    public void testSelectByBirth() throws ParseException {
        SqlSession sqlSession = SqlSessionUtil.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        Date birth = sdf.parse("1998-02-12");
        List<Student> students = mapper.selectByBirth(birth);
        for (Student s : students){
            System.out.println(s);
        }
        sqlSession.close();
    }

测试结果
在这里插入图片描述
可以看出,Date类型也可以缺省

具体都有哪些简单数据类型可以缺省呢?

经测试以下类型都可以缺省
byte short int long float double char
Byte Short Integer Long Float Double Character
String
java.util.Date
就不一一测试了

Map参数

接口中代码

 /***
     * 根据姓名和年龄来查询
     * @param map
     * @return
     */
    List<Student> selectByNameAndAge(Map<String, Object> map);

Sql映射文件中代码(缺省版本)

<select id="selectByNameAndAge" resultType="Student">
        select * from t_student where name = #{sname} and age = #{sage}
    </select>

注意#{}中的值必须是map中的键值
测试代码

@Test
    public void testSelectByNameAndAge(){
        SqlSession sqlSession = SqlSessionUtil.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        Map<String, Object> map = new HashMap<>();
        map.put("sname","张三");
        map.put("sage",20);
        List<Student> students = mapper.selectByNameAndAge(map);
        for(Student s : students)
            System.out.println(s);
        sqlSession.close();
    }

运行结果
在这里插入图片描述
得出结论Map也可以缺省

实体类参数

接口中代码

/***
     * 保存学生信息
     * @param s
     * @return
     */
    int insert(Student s);

Sql映射文件中代码(缺省parameterType)

<insert id="insert">
        insert into t_student values(null,#{name},#{age},#{height},#{birth},#{sex})
    </insert>

测试代码

@Test
    public void testInsert(){
        SqlSession sqlSession = SqlSessionUtil.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        Student student = new Student(null, "赵六", 18, 1.7, new Date(), '男');
        int count = mapper.insert(student);
        System.out.println(count);
        sqlSession.commit();
        sqlSession.close();
    }

运行结果
在这里插入图片描述
在这里插入图片描述
可以看出参数是实体类的时候也是可以缺省的

多个参数的情况

接口中代码

/***
     * 通过姓名和性别来进行查询
     * @param name
     * @param sex
     * @return
     */
    List<Student> selectByNameAndSex(String name,Character sex);

Sql映射文件中代码(缺省参数类型)

 <select id="selectByNameAndSex" resultType="Student">
        select * from t_student where name = #{name} and sex = #{sex}
    </select>

测试代码

@Test
    public void testSelectByNameAndSex(){
        SqlSession sqlSession = SqlSessionUtil.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        List<Student> students = mapper.selectByNameAndSex("张三", '男');
        for(Student s : students)
            System.out.println(s);
        sqlSession.close();
    }

测试结果
在这里插入图片描述

可以看出运行出现了问题,提醒参数name没有找到,并且提醒可用的参数为[arg1, arg0, param1, param2]
更改Sql映射文件中的代码为如下

<select id="selectByNameAndSex" resultType="Student">
        select * from t_student where name = #{arg0} and sex = #{arg1}
    </select>

再次运行结果为
在这里插入图片描述
将文件更改为

<select id="selectByNameAndSex" resultType="Student">
        select * from t_student where name = #{param1} and sex = #{param2}
    </select>

也可以通过

结论

可以看出当参数有多个的时候,MyBatis已经为我们起好了参数名,我们需要正确地使用MyBatis为我们准备好的参数名

arg0 是第⼀个参数

param1是第⼀个参数

arg1 是第⼆个参数

param2是第⼆个参数

实现原理:实际上在mybatis底层会创建⼀个map集合,以arg0/param1为key,以⽅法上的参数为 value
例如如下代码

Map<String,Object> map = new HashMap<>();
map.put("arg0", name);
map.put("arg1", sex);
map.put("param1", name);
map.put("param2", sex);
// 所以可以这样取值:#{arg0} #{arg1} #{param1} #{param2}
// 其本质就是#{map集合的key}

注意:使⽤mybatis3.4.2之前的版本时:要⽤#{0}和#{1}这种形式。

疑问:我们必须使用MyBatis为我们准备的参数吗? @Param注解登场


@Param注解(命名参数)

如果我们不想使用系统提供的参数名,我们可以自己规定吗?答案是肯定的

我们可以使用@Param来为参数起名字(也就是底层Map中参数对应的key值名)

实战

接口中的代码

/***
     * 通过姓名和性别来进行查询
     * @param name
     * @param sex
     * @return
     */
    List<Student> selectByNameAndSex(@Param(value = "name") String name,@Param("sex") Character sex);

value = 可以省略

更改Sql映射文件中的代码

<select id="selectByNameAndSex" resultType="Student">
        select * from t_student where name = #{name} and sex = #{sex}
    </select>

再次运行刚才的测试代码
运行结果
在这里插入图片描述
可以看出我们的@Param起作用了

核心:@Param(“这⾥填写的其实就是map集合的key”)
希望通过这篇文章大家都可以对MyBatis中的参数有一个更加深的理解

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/547725.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Java数据结构】——第十节(上).直接插入排序、希尔排序

作者简介&#xff1a;大家好&#xff0c;我是未央&#xff1b; 博客首页&#xff1a;未央.303 系列专栏&#xff1a;Java初阶数据结构 每日一句&#xff1a;人的一生&#xff0c;可以有所作为的时机只有一次&#xff0c;那就是现在&#xff01;&#xff01;&#xff01; 文章目…

iptables 防火墙(一)

目录 一&#xff1a;iptables概述 二&#xff1a;netfilter/iptables关系 三&#xff1a;四表五链 1.规则表和规则链的作用 2. 四表 3.五链 ​4.规则链之间的匹配顺序 &#xff08;1&#xff09;主机型防火墙 &#xff08;2&#xff09;网络型防火墙 5.规则链内的匹配…

考研复试刷题第十四天: 表达式树 【二叉树,表达式运算】

1.概念解释: 表达式树其实就是叶节点装树&#xff0c;其他节点装符号的二叉树。 2.题目部分 这道题一开始没理解它的意思&#xff0c;以后写题一定要理解题意之后再动手。尤其是看清楚注意事项。 我一开始拿到题目&#xff0c;以为会有这种情况就是说一个节点之下会有一遍没…

面了一个测试工程师要求月薪26K,总感觉他背了很多面试题...

最近有朋友去华为面试&#xff0c;面试前后进行了20天左右&#xff0c;包含4轮电话面试、1轮笔试、1轮主管视频面试、1轮hr视频面试。 据他所说&#xff0c;80%的人都会栽在第一轮面试&#xff0c;要不是他面试前做足准备&#xff0c;估计都坚持不完后面几轮面试。 其实&…

OpenLayers多图层切换显示隐藏,支持多个Layer同时显示和隐藏,以百度地图和高德地图为例实现vue的Layer图层管理组件

前言 OpenLayers默认并没有提供图层管理组件,实现起来也很简单,评论区里有同学提到了这个,必须立刻满足,这就着手区实现一个简单又强大的地图图层管理组件。 那么本章就专门讲一下在vue中如何使用ElementUI的下拉框做一个简单的图层管理组件。 话不多说,让我们直接开始吧…

CentOS中vim的使用

vim是我们linux中很经典的一款编译器&#xff0c;所以使用vim是我们在学习过程中必不可少的&#xff0c;我们下面说一下vim的使用和安装 在某些服务器上刚开始不一定时有vim的&#xff0c;或者是vim的版本比较老一点&#xff0c;所以这时我们就可以安装一下vim sudo yum -y i…

最简单的 Java 项目——Hello world(小白快速入门指南)

文章目录 最简单的 Java 项目——Hello world步骤1&#xff1a;新建 Java 项目步骤2&#xff1a;编写最简单的 Hello World 程序步骤3&#xff1a;测试 附录1、.iml文件&#xff08;iml是 intellij idea的工程配置文件&#xff0c;里面是当前project的一些配置信息。&#xff0…

免费开源PCB设计工具--KiCad安装,FreeCAD下载方法

中小企业在使用AD等工具时&#xff0c;会被律师函关照&#xff0c;下面介绍一款跨平台开源PCB设计工具KiCad 。本文仅介绍安装方法。 1. KiCad 简介 KiCad 一个跨平台的开源电子设计自动化套件。 KiCad EDA 是一款用于印刷电路板设计的开源自由软件&#xff0c;最初由法国人…

Shell编程——iptables防火墙

Shell编程——iptables防火墙 一、Linux包过滤防火墙1、Linux防火墙概述2、netfilter3、iptables4、netfilter/iptables关系 二、四表五链1、表链作用2、四表3、五链4、数据包到达防火墙时&#xff0c;规则表之间的优先顺序5、规则链之间的匹配顺序 三、iptables的安装四、ipta…

Swing简述

一、Swing概述 GUI&#xff08;图形用户界面&#xff09;为程序提供图形界面&#xff0c;它最初的设计目的是为程序员构建一个通用的GUI&#xff0c;使其能够在所有的平台上运行&#xff0c;但Java 1.0中基础类AWT&#xff08;抽象窗口工具箱&#xff09;并没有达到这个要求&a…

chatgpt赋能Python-python3_kafka

简介 Kafka是一个分布式的消息队列系统&#xff0c;由LinkedIn开源。它被设计成高性能、高吞吐量的消息传输系统&#xff0c;适用于分布式系统中的实时数据流处理。 Kafka的优势 在使用Kafka之前&#xff0c;我们需要考虑以下问题&#xff1a; 1.数据处理速度是否快速&…

黑客如何从零学起?

一、MYSQL5.7 MySQL是如今使用最多的数据库&#xff0c;是众多企业的首选&#xff0c;在未来几年都将被持续推动发展。 学习MySQL需注重实战操作&#xff0c;循序渐进地了解MySQL中的各项技术&#xff0c;这样才能在实际工作中的关键应用。 想进入网络安全行业&#xff0c; …

机器学习-2 线性回归

线性回归及最大熵模型 算法概述最小二乘法一元线性回归求解方程系数代价函数最小二乘法求解系数 多元线性回归举例 算法应用数据集介绍实现线性回归算法实现线性回归的算法流程最小二乘法的局限性 梯度下降法场景梯度下降算法&#xff08;Gradient Descent&#xff09;算法实例…

Electron,我与你,今天不谈技术谈感情!

目录 前言一、无知二、初见三、再见四、相遇五、行动总结 前言 今天不谈技术&#xff0c;谈谈我和 Electron 的缘分。可能有人觉得&#xff0c;或许有些人认为&#xff0c;和一个框架谈感情这不是疯了吗&#xff1f;但是&#xff0c;我相信每个开发者都会有同样的经历&#xf…

关于蒙特卡罗方法及其在信号处理中的应用

说明 最近想探讨一下毫米波雷达测量准确度及其改善的问题&#xff0c;这个话题下可供讨论的问题有很多&#xff0c;蒙特卡罗方法(或者说基于蒙特卡罗方法对测量准确度以及精度的评估)是其中之一&#xff0c;该方法是一个十分有效的工具&#xff0c;在科研(发paper)上也是不可少…

UE5 C++类如何读取Excel配置表?

UE5 插件开发指南 前言0 如何编写读取数据的结构体?1 如何读取数据?1.0 如何获取数据资产的路径?2 如何调用商店子系统来读取数据?前言 虚幻引擎兼容CSV和JSON格式的数据结构,这里的CSV是Excel表格的保存格式,如下图所示: 打开任意Excel表格,点击文件菜单,然后鼠标悬浮到…

一文2000字从0到1教你搭建有效的测试环境

作为软件测试行业的从业者&#xff0c;搭建测试环境一定是在工作中少不了的任务安排&#xff0c;那么如何有效的搭建测试环境&#xff0c;咱们今天和大家聊一聊。 首先大家要明白测试环境是为了完成软件测试工作所需要的硬件资源&#xff0c;软件资源&#xff0c;网络资源&…

计算机网络实验(ensp)-​实验2:PPP协议及PAP认证

目录 实验报告&#xff1a; 实验操作 1.建立网络拓扑图并开启设备 2.修改路由器名字 1.输入命名&#xff1a;sys 从用户视图切换到系统视图 2.输入命名&#xff1a;sysname 姓名 修改路由器名字 3.重复步骤1和2配置每台路由器 3.抓包 1.点击菜单栏的“数…

(转载)从0开始学matlab(第7天)—运算的优先级与内置函数

1运算的优先级 许多的数学运算写入一个表达式是非常平常的事。例如&#xff0c;考虑初速度为 0 的匀加速运动的位移表达式 distance 0.5 * accel * time ^ 2 这个表达式有二个乘法运算和一个幂运算。在这样的表达式中&#xff0c;知道运算的先后顺序是十分重要的。如果幂运算先…

Redis系统学习篇-02

目录 redis进阶使用 redis作为数据库、缓存的区别 缓存常见问题、面试回答思路 redis的持久化 参考文档(redis.net.cn)&#xff1a;Redis 教程_redis教程 系统学习01篇&#xff0c;总结起来就是&#xff0c;可以把redis想象成一个key/value的hashmap&#xff0c;只不过val…