网站首页 > java教程 正文
背景
作为一名后端开发攻城狮,您是否碰到过无法正确模糊查询到含有反斜杠\、百分号%、下划线_等关键字的内容? 比如想模糊查询一下文件信息表中,文件名称包含了下划线的文件信息都有哪些,却返回了表中所有的文件信息或包含了明显不匹配的文件信息。在实际的项目开发中,这种情况很普遍,开发的时候如果没有处理好,到了测试阶段,往往会被测试的同学提BUG单。
在SpringBoot应用中,遇到这种问题,大部分攻城狮们第一时间想到的可能就是写个工具类,然后在接口的代码中,对用户输入的关键字进行特殊符号的转义后,再提交到数据库去查询,这也不失为一种方法。但这种方式需要修改接口的代码,包括测试用例代码,稍显麻烦,那么,有没有更优雅、更省事的实现模糊查询含有特殊符号的内容的方式呢? 答案当然是有的,就是MyBatis提供的扩展接口:TypeHandler类型处理器。
1、创建特殊符号转义类型转换器类
实现TypeHandler接口,在setParameter方法(如果是继承BaseTypeHandler,则是在setNonNullParameter方法)中对特殊符号进行转义处理。
/*
* Copyright (c) Bruce.CH 2022-2025. All rights reserved.
*/
import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;
import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
/**
* 模糊查询字段特殊符号转义处理器
*
* @author Bruce.CH
* @since 2025年02月27日
*/
@MappedJdbcTypes(JdbcType.VARCHAR)
@MappedTypes(String.class)
public class EscapeTypeHandler extends BaseTypeHandler<String> {
@Override
public void setNonNullParameter(PreparedStatement ps, int i, String parameter, JdbcType jdbcType) throws SQLException {
// 对查询关键字中的特殊符号进行转义
String escapedParameter = parameter
.replace("\\", "\\\\") // 转义反斜杠
.replace("%", "\\%") // 转义百分号
.replace("_", "\\_"); // 转义下划线
ps.setString(i, escapedParameter);
}
@Override
public String getNullableResult(ResultSet rs, String columnName) throws SQLException {
// 查询结果不需要处理,原样返回即可
return rs.getString(columnName);
}
@Override
public String getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
// 查询结果不需要处理,原样返回即可
return rs.getString(columnIndex);
}
@Override
public String getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
// 查询结果不需要处理,原样返回即可
return cs.getString(columnIndex);
}
}
在上述代码中:
- @MappedJdbcTypes(JdbcType.VARCHAR) 和 @MappedTypes(String.class) 注解表示该类型处理器应用于 VARCHAR 类型的 JDBC 参数和 Java 的 String 类型。
- setNonNullParameter 方法中对传入的字符串参数中的 \、% 和 _ 进行转义,即在这些字符前加上 \。
2、在 Mapper XML 中使用
在 Mapper XML 文件的 查询SQL 语句中,使用特殊符号转义类型转换器类。
<select id="query" resultType="com.example.demo.vo.FileInfoVo">
SELECT id,file_name,file_type,file_size FROM tb_file
WHERE file_name LIKE concat('%', #{fileName, typeHandler=com.example.demo.typehandler.EscapeTypeHandler}, '%')
LIMIT 10
</select>
在上述 SQL 语句中,fileName参数使用了自定义的 EscapeTypeHandler类型处理器,这样在设置参数时,特殊符号如反斜杠、百分号、下划线等会被自动转义,这样就非常简单、优雅、正确的实现根据文件名称模糊查询符合条件的文件信息。
3、测试
输入下划线:查询文件名称带有下划线的文件信息
输入下百分号:查询文件名称带有百分号的文件信息
均能正确返回符合条件的文件信息。
4、总结
通过以上步骤,就可以在 Spring Boot 应用中使用 MyBatis 的类型转换器对查询关键字中的特殊符号进行转义处理。
猜你喜欢
- 2025-05-08 探索Java世界的新天地:JDK最新特性解读
- 2025-05-08 Java 15 新特性:文本块(java纯文本)
- 2025-05-08 贼好用的 Java 工具类库(java常用工具包)
- 2025-05-08 Java编程学习入门、Java语言学习、Java入门必看
- 2025-05-08 Android关于Groovy语言,你知道哪些?
- 2025-05-08 Java Web开发中常见漏洞及防护(java上传漏洞)
- 2025-05-08 轻量级的数据交换语言——JSON(数据交换和接口调用)
- 2025-05-08 Java程序语言之灵魂一(程序的灵魂算法)
- 2025-05-08 java正则表达式的使用(java正则怎么用)
- 2025-05-08 以后我准备告别String.format()了,因为它不够香!
你 发表评论:
欢迎- 05-08Hive-数据类型(hive数据类型和文件格式)
- 05-08SpringBoot系列Mybatis之ResultMap、ResultType返回结果使用姿势
- 05-08Linux shell变量&运算符(shell 命令中使用变量)
- 05-08详解Xss 及SpringBoot 防范Xss攻击(附全部代码)
- 05-08MyBatis-Plus码之重器 lambda 表达式使用指南,开发效率瞬间提升80%
- 05-08linux运维中特殊符号的应用与实践
- 05-08深入理解JAVA I/O系列一:File(java.io.fileinputstream)
- 05-08探索Java世界的新天地:JDK最新特性解读
- 最近发表
-
- Hive-数据类型(hive数据类型和文件格式)
- SpringBoot系列Mybatis之ResultMap、ResultType返回结果使用姿势
- Linux shell变量&运算符(shell 命令中使用变量)
- 详解Xss 及SpringBoot 防范Xss攻击(附全部代码)
- MyBatis-Plus码之重器 lambda 表达式使用指南,开发效率瞬间提升80%
- linux运维中特殊符号的应用与实践
- 深入理解JAVA I/O系列一:File(java.io.fileinputstream)
- 探索Java世界的新天地:JDK最新特性解读
- Java 15 新特性:文本块(java纯文本)
- 贼好用的 Java 工具类库(java常用工具包)
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)