我的编程空间,编程开发者的网络收藏夹
学习永远不晚

浅谈Mybatis传参类型如何确定

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

浅谈Mybatis传参类型如何确定

最近有小伙伴在讨论#{}与${}的区别时,有提到#{}是用字符串进行替换,就我个人的理解,它的主要作用是占位,最终替换的结果并不一定是字符串方式,比如我们传参类型是整形时,最终拼接的sql,传参讲道理也应该是整形,而不是字符串的方式

接下来我们来看一下,mapper接口中不同的参数类型,最终拼接sql中是如何进行替换的

I. 环境配置

我们使用SpringBoot + Mybatis + MySql来搭建实例demo

springboot: 2.2.0.RELEASE
mysql: 5.7.22

1. 项目配置


<dependencies>
    <dependency>
        <groupId>org.mybatis.spring.boot</groupId>
        <artifactId>mybatis-spring-boot-starter</artifactId>
        <version>2.2.0</version>
    </dependency>
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
    </dependency>
</dependencies>

核心的依赖mybatis-spring-boot-starter,至于版本选择,到mvn仓库中,找最新的
另外一个不可获取的就是db配置信息,appliaction.yml


spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/story?useUnicode=true&characterEncoding=UTF-8&useSSL=false&serverTimezone=Asia/Shanghai
    username: root
    password:

2. 数据库表

用于测试的数据库


CREATE TABLE `money` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(20) NOT NULL DEFAULT '' COMMENT '用户名',
  `money` int(26) NOT NULL DEFAULT '0' COMMENT '钱',
  `is_deleted` tinyint(1) NOT NULL DEFAULT '0',
  `create_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
  `update_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
  PRIMARY KEY (`id`),
  KEY `name` (`name`)
) ENGINE=InnoDB AUTO_INCREMENT=551 DEFAULT CHARSET=utf8mb4;

测试数据,主要是name字段,值为一个数字的字符串


INSERT INTO `money` (`id`, `name`, `money`, `is_deleted`, `create_at`, `update_at`)
VALUES
 (120, '120', 200, 0, '2021-05-24 20:04:39', '2021-09-27 19:21:40');

II. 传参类型确定

本文忽略掉mybatis中的po、mapper接口、xml文件的详情,有兴趣的小伙伴可以直接查看最下面的源码(或者查看之前的博文也可以)

1. 参数类型为整形

针对上面的case,定义一个根据name查询数据的接口,但是这个name参数类型为整数
mapper接口:



List<MoneyPo> queryByName(@Param("name") Integer name);

对应的xml文件如下


<select id="queryByName" resultMap="BaseResultMap">
    select * from money where `name` = #{name}
</select>

上面这个写法非常常见了,我们现在的问题就是,传参为整数,那么最终的sql是 name = 120 还是 name = '120'呢?
那么怎么确定最终生成的sql是啥样的呢?这里介绍一个直接输出mysql执行sql日志的方式
在mysql服务器上执行下面两个命令,开启sql执行日志


set global general_log = "ON";
show variables like 'general_log%';

当我们访问上面的接口之后,会发现最终发送给mysql的sql语句中,参数替换之后依然是整数


select * from money where `name` = 120

2. 指定jdbcType

在使用#{}, ${}时,有时也会看到除了参数之外,还会指定jdbcType,那么我们在xml中指定这个对最终的sql生成会有影响么?


<select id="queryByNameV2" resultMap="BaseResultMap">
    select * from money where `name` = #{name, jdbcType=VARCHAR} and 0=0
</select>

生成的sql如下


select * from money where `name` = 120 and 0=0

从实际的sql来看,这个jdbcType并没有影响最终的sql参数拼接,那它主要是干嘛用呢?(它主要适用于传入null时,类型转换可能出现的异常)

3. 传参类型为String

当我们传参类型为string时,最终的sql讲道理应该会带上引号



List<MoneyPo> queryByNameV3(@Param("name") String name);

对应的xml


<select id="queryByNameV3" resultMap="BaseResultMap">
    select * from money where `name` = #{name, jdbcType=VARCHAR} and 1=1
</select>

上面这个最终生成的sql如下
select * from money where `name` = '120' and 1=1

4. TypeHandler实现参数替换强制添加引号

看完上面几节,基本上可以有一个得出一个简单的推论(当然对不对则需要从源码上分析了)

sql参数替换,最终并不是简单使用字符串来替换,实际上是由参数java的参数类型决定,若java参数类型为字符串,拼接的sql为字符串格式;传参为整型,拼接的sql也是整数

那么问题来了,为什么要了解这个?

关键点在于索引失效的问题

比如本文实例中的name上添加了索引,当我们的sql是 select * from money where name = 120 会走不了索引,如果想走索引,要求传入的参数必须是字符串,不能出现隐式的类型转换

基于此,我们就有一个应用场景了,为了避免由于传参类型问题,导致走不了索引,我们希望name的传参,不管实际传入参数类型是什么,最终拼接的sql,都是字符串的格式;

我们借助自定义的TypeHandler来实现这个场景


@MappedTypes(value = {Long.class, Integer.class})
@MappedJdbcTypes(value = {JdbcType.CHAR, JdbcType.VARCHAR, JdbcType.LONGVARCHAR})
public class StrTypeHandler extends BaseTypeHandler<Object> {

    
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, Object parameter, JdbcType jdbcType) throws SQLException {
        ps.setString(i, String.valueOf(parameter));
    }

    
    @Override
    public Object getNullableResult(ResultSet rs, String columnName) throws SQLException {
        return rs.getString(columnName);
    }

    @Override
    public Object getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        return rs.getString(columnIndex);
    }

    @Override
    public Object getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        return cs.getString(columnIndex);
    }
}

然后在xml中,指定TypeHandler



List<MoneyPo> queryByNameV4(@Param("name") Integer name);

<select id="queryByNameV4" resultMap="BaseResultMap">
    select * from money where `name` = #{name, jdbcType=VARCHAR, typeHandler=com.git.hui.boot.mybatis.handler.StrTypeHandler} and 2=2
</select>

上面这种写法输出的sql就会携带上单引号,这样就可以从源头上解决传参类型不对,导致最终走不了索引的问题


select * from money where `name` = '120' and 2=2

5. 小结

本文通过一个简单的实例,来测试Mapper接口中,不同的参数类型,对最终的sql生成的影响

参数类型为整数时,最终的sql的参数替换也是整数(#{}并不是简单的字符串替换哦)
参数类型为字符串时,最终的sql参数替换,会自动携带'' (${}注意它不会自动带上单引号,需要自己手动添加)

当我们希望不管传参什么类型,最终生成的sql,都是字符串替换时,可以借助自定义的TypeHandler来实现,这样可以从源头上避免因为隐式类型转换导致走不了索引问题
最后疑问来了,上面的结论靠谱么?mybatis中最终的sql是在什么地方拼接的?这个sql拼接的流程是怎样的呢?
关于sql的拼接全流程,后续博文即将上线,我是一灰灰,走过路过的各位大佬帮忙点个赞、价格收藏、给个评价呗

 到此这篇关于Mybatis传参类型如何确定的文章就介绍到这了,更多相关Mybatis传参类型如何确定内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

浅谈Mybatis传参类型如何确定

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Mybatis传参类型怎么确定

这篇文章主要介绍Mybatis传参类型怎么确定,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!I. 环境配置我们使用SpringBoot + Mybatis + MySql来搭建实例demospringboot: 2.
2023-06-25

如何确定 PHP 函数参数的类型

php 语言中可通过下列方法确定函数参数类型:is_ 函数:使用 is_ 函数检查变量类型,如 is_int() 和 is_array()。类型提示:在函数参数中指定期望类型,使用 : 语法,如 function calculate_to
如何确定 PHP 函数参数的类型
2024-04-19

如何确定 Golang 函数的类型?

如何确定 go 函数的类型?有以下方法:使用 go 语言文档、使用代码编辑器、使用 typeof 运算符。如何确定 Go 函数的类型?在 Go 语言中,函数类型是一个重要的概念,它描述了函数的参数和返回值类型。确定函数的类型可以帮助我们理
如何确定 Golang 函数的类型?
2024-04-21

VB如何自定义类型参数

这篇文章主要介绍“VB如何自定义类型参数”,在日常操作中,相信很多人在VB如何自定义类型参数问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”VB如何自定义类型参数”的疑惑有所帮助!接下来,请跟着小编一起来学习吧
2023-06-17

当MyBatis 参数类型为String时如何解决

当MyBatis 参数类型为String时如何解决?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。1. 参数为String时的插值问题假设有下面一Dao接口方法p
2023-05-31

SpringCloud如何通过Feign传递List类型参数

小编给大家分享一下SpringCloud如何通过Feign传递List类型参数,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!通过Feign传递List类型参数首先
2023-06-29

C++ 函数的返回值类型如何确定?

函数的返回值类型声明了函数将返回的值类型,避免类型不匹配和错误。确定返回值类型要考虑函数目的、操作、调用代码和可重用性。可选择的基本类型、结构体、类、指针和引用作为返回值类型。C++ 函数的返回值类型在 C++ 中,函数的返回值类型声明了
C++ 函数的返回值类型如何确定?
2024-04-19

如何确定泛型类型在运行时是否“可比较”?

问题内容我想编写一个通用的 equals 方法,其工作原理如下:func equals[T any](a, b T) bool {if hasEqualsMethod(T) {return a.Equals(b)else if isCo
如何确定泛型类型在运行时是否“可比较”?
2024-02-05

浅谈数据库日期类型字段设计应该如何选择

当设计一个产品,其中很多地方要把日期类型保存到数据库中,如果产品有兼容不同数据库产品的需求,那么,应当怎样设计呢?当然,首先想到的是,使用数据库的 Date 或 DateTime 类型,可是看看不同数据库这些类型间的区别吧,真让人望而止步
2022-08-12

PHP7中的Type Hinting特性:如何明确函数的参数类型以避免错误和不确定性?

PHP7中的Type Hinting特性:如何明确函数的参数类型以避免错误和不确定性?引言:在开发过程中,我们经常遇到函数传参的问题。有时候,由于函数参数类型不明确,导致出现错误和不确定性。为了解决这个问题,PHP7引入了Type Hint
2023-10-22

如何使用mybatis自定义日期类型转换器

本篇内容主要讲解“如何使用mybatis自定义日期类型转换器”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何使用mybatis自定义日期类型转换器”吧!前言使用mybatis中的typeHan
2023-06-29

PHP7中的Type Hinting特性:如何明确函数的参数类型?

PHP7中引入了Type Hinting特性,它允许开发者在函数声明中明确指定参数的类型。在开发过程中,我们经常会遇到参数类型错误导致的bug,Type Hinting可以帮助我们更早地发现这些错误,并在编译阶段提供更好的类型安全。Type
2023-10-25

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录