Mybatis 拦截器实现 Like 通配符转义

Mybatis 拦截器实现 Like 通配符转义

mysql中like查询通配符问题描述

如果在MySQL中使用如下查询,将检索出全部数据

select * from t_user where name like '%%%';
select * from t_user where name like '%_%';
select * from t_user where name like concat('%','%','%');
select * from t_user where name like concat('%','_','%');

所以我们要对其做一个转义处理
拦截目标为 like 查询语句

定义mybatis拦截器

import com.github.pagehelper.util.MetaObjectUtil;
import org.apache.ibatis.cache.CacheKey;
import org.apache.ibatis.executor.Executor;
import org.apache.ibatis.mapping.*;
import org.apache.ibatis.plugin.*;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.RowBounds;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;@Intercepts({@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}), @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class, CacheKey.class, BoundSql.class}),})
public class LikeStringEscapeInterceptor implements Interceptor {private static final Pattern LIKE_PARAM_PATTERN = Pattern.compile("like\\s+['\"%_]*\\?", Pattern.CASE_INSENSITIVE);private static final Pattern LIKE_CONCAT_PARAM_PATTERN = Pattern.compile("like\\s+concat\\s*\\(\\s*'%'\\s*,\\s*\\?,\\s*'%'\\s*\\)", Pattern.CASE_INSENSITIVE);@Overridepublic Object plugin(Object target) {return Plugin.wrap(target, this);}@Overridepublic Object intercept(Invocation invocation) throws Throwable {Object[] args = invocation.getArgs();MappedStatement ms = (MappedStatement) args[0];Object parameter = args[1];RowBounds rowBounds = (RowBounds) args[2];ResultHandler resultHandler = (ResultHandler) args[3];Executor executor = (Executor) invocation.getTarget();CacheKey cacheKey;BoundSql boundSql;if (args.length == 4) {boundSql = ms.getBoundSql(parameter);cacheKey = executor.createCacheKey(ms, parameter, rowBounds, boundSql);} else {cacheKey = (CacheKey) args[4];boundSql = (BoundSql) args[5];}SqlCommandType sqlCommandType = ms.getSqlCommandType();StatementType statementType = ms.getStatementType();if (sqlCommandType == SqlCommandType.SELECT && statementType == StatementType.PREPARED) {escapeParameterIfContainingLike(boundSql);return executor.query(ms, parameter, rowBounds, resultHandler, cacheKey, boundSql);}return invocation.proceed();}void escapeParameterIfContainingLike(BoundSql boundSql) {if (boundSql == null) {return;}String prepareSql = boundSql.getSql();List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();Map<Integer, Boolean> position = findLikeParam(prepareSql);if (position == null || position.size() == 0) {return;}Map<ParameterMapping, Boolean> likeParameterMappings = new LinkedHashMap<>();MetaObject metaObject = MetaObjectUtil.forObject(boundSql.getParameterObject());for (int i = 0; i < parameterMappings.size(); i++) {ParameterMapping pm = parameterMappings.get(i);String property = pm.getProperty();if (metaObject.hasGetter(property)) {boundSql.setAdditionalParameter(property, metaObject.getValue(property));if (position.containsKey(i)) {likeParameterMappings.put(pm, position.get(i));}}}delegateMetaParameterForEscape(boundSql, likeParameterMappings);}void delegateMetaParameterForEscape(BoundSql boundSql, Map<ParameterMapping, Boolean> likeParameterMappings) {for (ParameterMapping mapping : likeParameterMappings.keySet()) {String property = mapping.getProperty();MetaObject metaObject = MetaObjectUtil.forObject(boundSql.getParameterObject());Object value = metaObject.getValue(property);if (value instanceof String) {boundSql.setAdditionalParameter(property, escapeLike((String) value, likeParameterMappings.get(mapping)));}}}String escapeLike(String value, Boolean hasConcat) {if (value != null) {if (hasConcat) {return value.replaceAll("%", "\\\\%");}//去除首尾%如果有的话if (value.startsWith("%")) {value = value.substring(1);}if (value.endsWith("%")) {value = value.substring(0, value.length() - 1);}if (StringUtils.isEmpty(value)) {return null;}return "%" + value.replaceAll("%", "\\\\%") + "%";}return null;}Map<Integer, Boolean> findLikeParam(String prepareSql) {Matcher matcher = LIKE_PARAM_PATTERN.matcher(prepareSql);Map<Integer, Boolean> indexes = new LinkedHashMap<>();while (matcher.find()) {int start = matcher.start();int index = StringUtils.countMatches(prepareSql.substring(0, start), "?");indexes.put(index, false);}matcher = LIKE_CONCAT_PARAM_PATTERN.matcher(prepareSql);while (matcher.find()) {int start = matcher.start();int index = StringUtils.countMatches(prepareSql.substring(0, start), "?");indexes.put(index, true);}return indexes;}
}

使用拦截器

    @Beanpublic LikeStringEscapeInterceptor likeStringEscapeInterceptor() {return new LikeStringEscapeInterceptor();}

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

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

相关文章

java.lang.UnsupportedOperationException异常解决

在执行如下代码时&#xff0c;发现当apps.add("...");代码执行时&#xff0c;会报java.lang.UnsupportedOperationException错误 List<String> apps Arrays.asList("...");apps.add("..."); 问题出现的原因如下&#xff1a; 1、ArrayLi…

K8S学习指南(17)-k8s核心对象CronJob

文章目录 前言什么是CronJob&#xff1f;示例演示步骤1&#xff1a;创建CronJob步骤2&#xff1a;定义任务模板步骤3&#xff1a;部署CronJob步骤4&#xff1a;监视CronJob的执行 总结 前言 Kubernetes&#xff08;简称K8s&#xff09;是一种用于自动部署、扩展和管理容器化应…

STM32 寄存器配置笔记——I2C 读写AT24C02 EEPROM

一、简介 本文主要介绍STM32F10xx系列如何使用软件模拟I2C总线读写AT24C02的EEPROM数据。 二、概述 I2C协议是一种用于同步、半双工、串行总线(由单片机时钟线、单数据交换器数据线组成)上的协议。规定了总线空闲状态、起始条件、停止条件、数据有效性、字节格式、响应确认信号…

MES系统物料计划协同:全面解析与应用

一、MES系统物料计划协同概述 MES系统物料计划协同是指通过MES系统对物料计划进行统一管理和协调&#xff0c;确保生产计划的顺利进行。通过将物料需求、采购、库存、生产和配送等环节进行有效集成&#xff0c;实现供应链的优化。这种协同方式有助于提高供应链的透明度和协同性…

焊盘:十字连接VS全覆盖 铺铜

在铺铜规则中&#xff0c;焊盘连接方式有两种&#xff1a; 十字连接 优点&#xff1a;较好焊接&#xff1a;因铺铜面积减少&#xff0c;温度下降速度降低&#xff0c;较好焊接&#xff0c;不易虚焊。 缺点&#xff1a;载流能力较弱&#xff1a;铺铜面积↓ → 载流能力↓全连接…

Python: list查找元素操作

not in 在Python中&#xff0c;not in 是一个逻辑运算符&#xff0c;用于检查一个元素是否不在一个列表&#xff08;list&#xff09;中。 如果元素不在列表中&#xff0c;not in 返回 True&#xff0c;否则返回 False。 语法&#xff1a; element not in list 其中&#…

Arrays类

一&#xff0c;操作数组的工具类 方法名 说明 public static String toString​(类型[] arr) 返回数组的内容 public static int[] copyOfRange​(类型[] arr, 起始索引, 结束索引) 拷贝数组&#xff08;指定范围&#xff09; public static copyOf​(类型[] arr, int n…

Leetcode—118.杨辉三角【简单】

2023每日刷题&#xff08;六十&#xff09; Leetcode—118.杨辉三角 实现代码 class Solution { public:vector<vector<int>> generate(int numRows) {vector<vector<int>> ans(numRows);for(int i 0; i < numRows; i) {ans[i].resize(i 1);ans…

算法训练营Day14

#Java #二叉树层次遍历 #反转二叉树 开源学习资料 二叉树的层次遍历&#xff1a;力扣题目链接 二叉树的层次遍历很好理解&#xff1a; 就是从根结点一层一层地往下遍历&#xff08;同一层&#xff0c;从左到右&#xff09;&#xff1a; 迭代的方式很好理解&#xff1a;就是…

用实例域代替序数

在Java中&#xff0c;枚举类型的ordinal()方法返回枚举常量的序数&#xff08;即其在枚举声明中的位置&#xff09;。在某些情况下&#xff0c;使用实例域&#xff08;instance field&#xff09;代替序数可能更加安全和易读。以下是一个示例&#xff0c;演示如何使用实例域代替…

mysql CREATE DATABASE

DROP DATABASE IF EXISTS zengwenfeng;CREATE DATABASE zengwenfeng DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;SET NAMES utf8mb4; SET FOREIGN_KEY_CHECKS 0;USE zengwenfeng; 脚本天天少这些&#xff0c;天天找这段&#xff01;

computed 和 watch 的奇妙世界:让数据驱动你的 Vue 应用(下)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

TestSSLServer4.exe工具使用方法简单介绍(查SSL的加密版本SSL3或是TLS1.2)

一、工具使用方法介绍 工具使用方法参照&#xff1a;http://www.bolet.org/TestSSLServer/ 全篇英文看不懂&#xff0c;翻译了下&#xff0c;能用到的简单介绍如下&#xff1a; 将下载的TestSSLServer4.exe工具放到桌面上&#xff0c;CMD命令行进入到桌面目录&#xff0c;执…

解决Requires: socat异常

安装mariaDB10报错&#xff1a; Error: Package: galera-4-26.4.16-1.el7.centos.x86_64 (mariadb)Requires: socat Error: Package: galera-4-26.4.16-1.el7.centos.x86_64 (mariadb)Requires: libboost_program_options-mt.so.1.53.0()(64bit)You could try using --skip-br…

从 0 开始创建 SpringBoot 项目

从 0 开始创建 SpringBoot 项目 从 0 开始创建 SpringBoot 项目环境准备创建项目项目目录结构及说明编写代码参考 从 0 开始创建 SpringBoot 项目 环境准备 操作系统&#xff1a;Windows 10IDE&#xff1a;IntelliJ IDEA 2023.3.1Java 版本&#xff1a;jdk1.8 工具网盘链接&…

俄罗斯军方计划用 Astra Linux 取代 Windows!

网络安全正在改变全球化的面貌&#xff0c;各国政府为了防范外国的间谍和破坏活动&#xff0c;正积极发展自己的技术。在这一趋势下&#xff0c;俄罗斯军方已经开始用 Linux 发行版 Astra Linux 替换 Windows 系统。 如何提高Linux系统安全性&#xff1f;提升Linux安全的关键策…

垃圾收集器及内存分配

目录 垃圾收集器种类 HotSpot虚拟机所包含的收集器 垃圾收集器部分源码 垃圾收集器后台日志参数说明与配对关系 1、串行垃圾收集器 串行垃圾收集器运行示意图 1&#xff09;、编写测试代码 2&#xff09;、设置垃圾回收为串行收集器 3&#xff09;、启动程序&#xff…

Flink 数据集类型

现实世界中&#xff0c;所有的数据都是以流式的形态产生的&#xff0c;不管是哪里产生的数据&#xff0c;在产生的过程中都是一条条地生成&#xff0c;最后经过了存储和转换处理&#xff0c;形成了各种类型的数据集。如下图所示&#xff0c;根据现实的数据产生方式和数据产生是…

基于JavaWeb+SSM+Vue微信小程序的移动学习平台系统的设计和实现

基于JavaWebSSMVue微信小程序的移动学习平台系统的设计和实现 源码获取入口Lun文目录前言主要技术系统设计功能截图订阅经典源码专栏Java项目精品实战案例《500套》 源码获取 源码获取入口 Lun文目录 第1章 绪论 1 1.1 课题背景 1 1.2 课题意义 1 1.3 研究内容 2 第2章 开发环…

【基础篇】1.1 认识STM32(二)

3.3 VREF/VREF-引脚 VREF和VREF-是STM32中用于提供参考电压的引脚。如下图&#xff1a; VREF引脚可以连接一个单独的外部参考电压&#xff0c;范围在2.0V&#xff5e;VDDA&#xff0c;但不能超过VDDA&#xff0c;否则就超过了模拟器件的最大供电电压。在100引脚的封装中&#…