在 Java 的 MyBatis 框架中,# 和 $ 的区别

一、基本语法

xml 复制代码
<select id="getUserByName" resultType="User">
    SELECT * FROM user WHERE username = #{name}
</select>
xml 复制代码
<select id="getUserByTable" resultType="User">
    SELECT * FROM ${table} WHERE id = #{id}
</select>

二、区别详解

特性 # 占位符 $ 占位符
替换方式 预编译 ,使用 PreparedStatement 参数 直接字符串替换
SQL 安全性 安全(防止 SQL 注入) 不安全(容易被 SQL 注入)
使用场景 绝大多数参数替换,如 WHERE 条件中的值 表名、列名、排序字段等结构性元素
示例 WHERE name = #{name}WHERE name = ?,再将参数安全绑定为 admin ORDER BY ${column}ORDER BY username

三、举个例子理解差异

1. 使用 #

xml 复制代码
<select id="findUser" resultType="User">
    SELECT * FROM user WHERE username = #{username}
</select>

传入 username = "admin"

生成的 SQL 为:

sql 复制代码
SELECT * FROM user WHERE username = ?

然后通过 JDBC 的 PreparedStatementadmin 安全地绑定进去。

可以防止 SQL 注入,因为参数不会直接拼接进 SQL。


2. 使用 $

xml 复制代码
<select id="findUser" resultType="User">
    SELECT * FROM user WHERE username = '${username}'
</select>

传入 username = "admin' OR '1'='1"

生成的 SQL 为:

sql 复制代码
SELECT * FROM user WHERE username = 'admin' OR '1'='1'

❌ 这样就变成了 SQL 注入,永远返回真,导致严重安全漏洞。


四、典型使用场景

场景 使用方式 原因
传递查询条件、值 #{} 安全,防注入
动态表名、字段名 ${} 结构性语法,不能预编译,只能拼接
排序字段 ${} 例如 ORDER BY ${sortColumn}
SQL 片段(比如动态列名) ${} 用于构建 SELECT 字段列表等

五、总结口诀

  • 凡是值,尽量用 #{}
  • 凡是结构(表名、列名、排序字段),才考虑 ${},但一定要控制输入!

六、一个防坑建议

如果确实要使用 ${}(比如动态列名),一定要做白名单过滤

java 复制代码
// 假设只允许排序字段为 id 或 name
if (!"id".equals(sortColumn) && !"name".equals(sortColumn)) {
    throw new IllegalArgumentException("非法字段");
}

否则就可能被用户输入类似 "id; DROP TABLE user" 导致严重后果。

相关推荐
麦聪聊数据17 小时前
LiveOps事故零容忍:游戏行业数据库的细粒度权限管控与审计实践
运维·数据库·后端·sql
shepherd12617 小时前
深度剖析SkyWalking:从内核原理到生产级全链路监控实战
分布式·后端·skywalking
橘子师兄17 小时前
C++AI大模型接入SDK—Genimi接入封装
c++·人工智能·后端
码农水水17 小时前
大疆Java面试被问:使用Async-profiler进行CPU热点分析和火焰图解读
java·开发语言·jvm·数据结构·后端·面试·职场和发展
ahauedu17 小时前
SpringBoot 3.5.10引入springdoc-openapi-starter-webmvc-ui版本
java·spring boot·后端
未来龙皇小蓝18 小时前
Spring内置常见线程池配置及相关概念
java·后端·spring·系统架构
新缸中之脑18 小时前
Google:Rust实战评估
开发语言·后端·rust
一 乐18 小时前
在线考试|基于springboot + vue在线考试系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端·课程设计
女王大人万岁18 小时前
Go标准库 path 详解
服务器·开发语言·后端·golang
qq_124987075318 小时前
基于spring boot的调查问卷系统的设计与实现(源码+论文+部署+安装)
java·vue.js·spring boot·后端·spring·毕业设计·计算机毕业设计