Mybatis的mapper文件中#和$的区别示例解析

2025-12-08 20:50

本文主要是介绍Mybatis的mapper文件中#和$的区别示例解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Mybatis的mapper文件中#和$的区别示例解析》MyBatis的mapper文件中,#{}和${}是两种参数占位符,核心差异在于参数解析方式、SQL注入风险、适用场景,以下从底层原理、使用场...

MyBatis 中 mapper 文件里 #{} 与 ${} 的核心区别

MyBatis 的 mapper 文件中,#{}${}是两种参数占位符,核心差异在于参数解析方式、SQL 注入风险、适用场景,以下从底层原理、使用场景、示例等维度详细解析:

一、核心区别对比表

特性#{}${}
解析方式预编译处理:MyBatis 将#{}替换为?,通过 PreparedStatement 设置参数(setXxx()字符串替换:直接将参数拼接到 SQL 语句中,无预编译
SQL 注入风险安全(参数会被自动转义)危险(直接拼接,易触发 SQL 注入)
参数类型处理自动识别参数类型,自动加引号(如字符串参数拼接后为'value'不处理类型,直接拼接(字符串需手动加引号)
适用场景常规参数传递(列值、条件值)动态表名、列名、排序字段等 SQL 语法部分
性能预编译可复用执行计划,性能高每次拼接生成新 SQL,性能低

二、底层原理与示例

1.#{}:预编译占位符(推荐)

原理

MyBatis 解析#{}时,会将其替换为 JDBC 的?占位符,生成PreparedStatement,并通过setInt()/setString()等方法设置参数,参数会被自动转义(如特殊字符'会被处理为\'),从根本上避免 SQL 注入。

示例

Mapper 文件

<select id="getUserById" resultType="User">
    SELECT id, name, age FROM user WHERE id = #{id}
</select>

实际执行的 SQL(参数 id=1)

-- MyBatis生成PreparedStatement,参数通过setInt(1, 1)设置
SELECT id, name, age FROM user WHERE id = ?

字符串参数示例(name="张三")

<select id="getUserByName" resultType="User">
    SELECT id, name, age FROM user WHERE name = #{name}
</select>

实际执行:参数会被自动加引号,等价于WHERE name = '张三',若参数含特殊字符(如张三' OR 1=1),会被转义为'张三\' OR 1=1',避免注入。

2.${}:字符串拼接(慎用)

原理

MyBatis 直接将${}替换为参数的原始字符串,无任何转义处理,相当于 “硬拼接” SQL,因此存在严重的 SQL 注入风险,但可用于动态指定 SQL 语法的部分(如表名、列名)。

示例

动态表名(按年月分表)

<select id="getUserByMonth" resultType="User">
    SELECT id, name FROM user_${month} WHERE id = #{id}
</select>

参数 month="202512&quojavascriptt;,id=1 时,实际执行 SQL

SELECT id, name FROM use编程r_202512 WHERE id = ?

风险示例(SQL 注入):若使用${}接收用户输入的排序字段:

<select id="getUserList" resultType="User">
    SELECT id, name FROM user ORDER BY ${sortField}
</select>

当用户传入sortField="id; DROP TABLE user;"时,拼接后的 SQL 为:

SELECT id, name FROM user ORDER BY id; DROP TABLE user;

直接导致表被删除,风险极高。

三、关键使用原则

1. 优先使用#{}

所有常规参数(如查询条件、插入 / 更新的列值)必须用#{},杜绝 SQL 注入风险,同时享受预编译的性能优势。

2. 仅在必要时使用${}

仅当需要动态指定 SQL 语法元素时使用${},且必须做严格的参数校验:

// 对${}参数做白名单校验(示例:仅允许指js定的排序字段)
public List<User> getUserList(String sortField) {
    // 白名单
    List<String> allowedFields = Arrays.asList("id", "name", "age");
    if (!allowedFields.contains(sortField)) {
        sortField = "id"; // 默认值,防止注入
    }
    return userMapper.getUserList(sortField);
}

3. 特殊场景的兼容处理

模糊查询#{}可配合CONCAT使用(推荐),而非${}

<!-- 正确:避免注入 -->
SELECT * FROM user WHERE name LIKE CONCAT('%', #{name}, '%')
<!-- 错误:易注入 -->
SELECT * FROM user WHERE name LIKE '%${name}%'

批量插入:MyBatis 的foreach标签中,集合元素用#{}

<insert id="batchInsert">
    INSERT INhttp://www.chinasem.cnTO user (name, age) VALUES
    <foreach collection="list" item="item" separator=",">
        (#{item.name}, #{item.age})
    </foreach&g编程t;
</insert>

四、总结

场景推荐用法注意事项
传递列值 / 条件值#{}自动转义,无注入风险
动态表名 / 列名 / 排序${} + 白名单必须校验参数,限制取值范围
模糊查询#{} + CONCAT避免直接拼接%${}%

核心记住:#{}是 “安全的预编译”,${}是 “危险的字符串拼接”,开发中除非明确需要动态拼接 SQL 语法,否则一律使用#{}

到此这篇关于Mybatis的mapper文件中#和$的区别的文章就介绍到这了,更多相关mybatis #和$区别内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程China编程(www.chinasem.cn)!

这篇关于Mybatis的mapper文件中#和$的区别示例解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1156394

相关文章

HTML5的input标签的`type`属性值详解和代码示例

《HTML5的input标签的`type`属性值详解和代码示例》HTML5的`input`标签提供了多种`type`属性值,用于创建不同类型的输入控件,满足用户输入的多样化需求,从文本输入、密码输入、... 目录一、引言二、文本类输入类型2.1 text2.2 password2.3 textarea(严格

Agent开发核心技术解析以及现代Agent架构设计

《Agent开发核心技术解析以及现代Agent架构设计》在人工智能领域,Agent并非一个全新的概念,但在大模型时代,它被赋予了全新的生命力,简单来说,Agent是一个能够自主感知环境、理解任务、制定... 目录一、回归本源:到底什么是Agent?二、核心链路拆解:Agent的"大脑"与"四肢"1. 规划模

MyBatis-Plus逻辑删除实现过程

《MyBatis-Plus逻辑删除实现过程》本文介绍了MyBatis-Plus如何实现逻辑删除功能,包括自动填充字段、配置与实现步骤、常见应用场景,并展示了如何使用remove方法进行逻辑删除,逻辑删... 目录1. 逻辑删除的必要性编程1.1 逻辑删除的定义1.2 逻辑删php除的优点1.3 适用场景2.

MySQL字符串转数值的方法全解析

《MySQL字符串转数值的方法全解析》在MySQL开发中,字符串与数值的转换是高频操作,本文从隐式转换原理、显式转换方法、典型场景案例、风险防控四个维度系统梳理,助您精准掌握这一核心技能,需要的朋友可... 目录一、隐式转换:自动但需警惕的&ld编程quo;双刃剑”二、显式转换:三大核心方法详解三、典型场景

MySQL中between and的基本用法、范围查询示例详解

《MySQL中betweenand的基本用法、范围查询示例详解》BETWEENAND操作符在MySQL中用于选择在两个值之间的数据,包括边界值,它支持数值和日期类型,示例展示了如何使用BETWEEN... 目录一、between and语法二、使用示例2.1、betwphpeen and数值查询2.2、be

python中的flask_sqlalchemy的使用及示例详解

《python中的flask_sqlalchemy的使用及示例详解》文章主要介绍了在使用SQLAlchemy创建模型实例时,通过元类动态创建实例的方式,并说明了如何在实例化时执行__init__方法,... 目录@orm.reconstructorSQLAlchemy的回滚关联其他模型数据库基本操作将数据添

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

JAVA项目swing转javafx语法规则以及示例代码

《JAVA项目swing转javafx语法规则以及示例代码》:本文主要介绍JAVA项目swing转javafx语法规则以及示例代码的相关资料,文中详细讲解了主类继承、窗口创建、布局管理、控件替换、... 目录最常用的“一行换一行”速查表(直接全局替换)实际转换示例(JFramejs → JavaFX)迁移建

Spring Boot Interceptor的原理、配置、顺序控制及与Filter的关键区别对比分析

《SpringBootInterceptor的原理、配置、顺序控制及与Filter的关键区别对比分析》本文主要介绍了SpringBoot中的拦截器(Interceptor)及其与过滤器(Filt... 目录前言一、核心功能二、拦截器的实现2.1 定义自定义拦截器2.2 注册拦截器三、多拦截器的执行顺序四、过

MyBatis配置文件中最常用的设置

《MyBatis配置文件中最常用的设置》文章主要介绍了MyBatis配置的优化方法,包括引用外部的properties配置文件、配置外置以实现环境解耦、配置文件中最常用的6个核心设置以及三种常用的Ma... 目录MyBATis配置优化mybatis的配置中引用外部的propertis配置文件⚠️ 注意事项X