MyBatis-Plus模糊查詢特殊字符串轉(zhuǎn)義的實現(xiàn)
問題描述
使用MyBatis中的模糊查詢時,當(dāng)查詢關(guān)鍵字中包括有_、\、%時,查詢關(guān)鍵字失效。例如需要查詢出名稱帶下劃線的用戶,點擊查詢后卻查出了全部用戶。
解決方案
MyBatis-Plus會直接將用戶的輸入拼接到查詢語句中,例如%_%,但我們實際需要的是%\_%,為了解決這個問題可以使用一個MyBatis-Plus的全局?jǐn)r截器對特殊字符進(jìn)行轉(zhuǎn)義。下面是攔截器的源代碼:
/**
* 特殊字符轉(zhuǎn)換 like 查詢
* 參數(shù)中的特殊查詢字符 _ % \
* 注意事項 :
* 1. 必須是在 分頁攔截器之前執(zhí)行 [注意攔截器的順序]
* 2. 當(dāng)數(shù)據(jù)庫字段排序規(guī)則為utf8_unicode_ci時,對于特殊符號的模糊查詢失效,
* 所以,必須設(shè)置數(shù)據(jù)庫字段排序規(guī)則為utf8_general_ci或utf8mb4_general_ci
*
*/
@Intercepts(@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class,
RowBounds.class, ResultHandler.class}))
public class SpecialCharacterConversionLikeInnerInterceptor implements Interceptor {
@Override
public Object intercept(Invocation invocation) throws Throwable {
// 攔截sql
Object[] args = invocation.getArgs();
MappedStatement statement = (MappedStatement) args[0];
Object parameterObject = args[1];
BoundSql boundSql = statement.getBoundSql(parameterObject);
String sql = boundSql.getSql();
// 處理特殊字符
modifyLikeSql(sql, parameterObject, boundSql);
// 返回
return invocation.proceed();
}
@Override
public Object plugin(Object target) {
return Plugin.wrap(target, this);
}
@Override
public void setProperties(Properties properties) {
}
public String modifyLikeSql(String sql, Object parameterObject, BoundSql boundSql) {
if (parameterObject instanceof HashMap) {
} else {
return sql;
}
if (!sql.toLowerCase().contains(" like ") || !sql.toLowerCase().contains("?")) {
return sql;
}
// 獲取關(guān)鍵字的個數(shù)(去重)
String[] strList = sql.split("\\?");
Set<String> keyNames = new HashSet<>();
for (int i = 0; i < strList.length; i++) {
if (strList[i].toLowerCase().contains(" like ")) {
String keyName = boundSql.getParameterMappings().get(i).getProperty();
keyNames.add(keyName);
}
}
// 對關(guān)鍵字進(jìn)行特殊字符“清洗”,如果有特殊字符的,在特殊字符前添加轉(zhuǎn)義字符(\)
for (String keyName : keyNames) {
HashMap parameter = (HashMap) parameterObject;
if (keyName.contains("ew.paramNameValuePairs.") && sql.toLowerCase().contains(" like ?")) {
// 第一種情況:在業(yè)務(wù)層進(jìn)行條件構(gòu)造產(chǎn)生的模糊查詢關(guān)鍵字
AbstractWrapper wrapper = (AbstractWrapper) parameter.get("ew");
parameter = (HashMap) wrapper.getParamNameValuePairs();
String[] keyList = keyName.split("\\.");
// ew.paramNameValuePairs.MPGENVAL1,截取字符串之后,獲取第三個,即為參數(shù)名
Object a = parameter.get(keyList[2]);
if (a instanceof String && (a.toString().contains("_") || a.toString().contains("\\") || a.toString()
.contains("%"))) {
parameter.put(keyList[2],
"%" + escapeChar(a.toString().substring(1, a.toString().length() - 1)) + "%");
}
} else if (!keyName.contains("ew.paramNameValuePairs.") && sql.toLowerCase().contains(" like ?")) {
// 第二種情況:未使用條件構(gòu)造器,但是在service層進(jìn)行了查詢關(guān)鍵字與模糊查詢符`%`手動拼接
Object a = parameter.get(keyName);
if (a instanceof String && (a.toString().contains("_") || a.toString().contains("\\") || a.toString()
.contains("%"))) {
parameter.put(keyName,
"%" + escapeChar(a.toString().substring(1, a.toString().length() - 1)) + "%");
}
} else {
// 第三種情況:在Mapper類的注解SQL中進(jìn)行了模糊查詢的拼接
updateValue(parameter, keyName);
}
}
return sql;
}
public String escapeChar(String str) {
if (!StringUtils.isEmpty(str)) {
str = str.replaceAll("\\\\", "\\\\\\\\");
str = str.replaceAll("_", "\\\\_");
str = str.replaceAll("%", "\\\\%");
}
return str;
}
/**
* 從參數(shù)字典中取值
* 取出keyName對應(yīng)的value
*
* @param parameter 參數(shù)字典
* @param keyName key名稱
* @return value
*/
private void updateValue(HashMap parameter, String keyName) {
String v = "";
List<String> sub_keys = Arrays.asList(keyName.split("\\."));
if (sub_keys.size() == 1 && parameter.containsKey(keyName)) {
v = parameter.get(keyName).toString();
if ((v.contains("_") || v.contains("\\") || v.contains("%"))) {
parameter.put(keyName, escapeChar(v));
}
} else if (sub_keys.size() == 2 && parameter.containsKey(sub_keys.get(0))) {
HashMap bean = (HashMap) BeanUtil.beanToMap(parameter.get(sub_keys.get(0)));
if (bean.containsKey(sub_keys.get(1))) {
v = (String) bean.get(sub_keys.get(1));
if ((v.contains("_") || v.contains("\\") || v.contains("%"))) {
v = escapeChar(v);
}
bean.put(sub_keys.get(1), v);
parameter.put(sub_keys.get(0), bean);
}
} else {
// 暫不支持復(fù)雜的Bean
return;
}
}
}
該代碼參考互聯(lián)網(wǎng)代碼,并對第三種情況進(jìn)行了擴(kuò)展,目前可以支持自定義類參數(shù)。
之后將攔截器添加到配置類即可
@Configuration
public class MybatisPlusConfig {
/**
* 模糊查詢攔截器
*
* @return
*/
@Bean
public SpecialCharacterConversionLikeInnerInterceptor myInterceptor() {
return new SpecialCharacterConversionLikeInnerInterceptor();
}
}參考連接:
https://www.jianshu.com/p/f4d3e6ffeee8
到此這篇關(guān)于MyBatis-Plus模糊查詢特殊字符串轉(zhuǎn)義的實現(xiàn)的文章就介紹到這了,更多相關(guān)MyBatisPlus字符串轉(zhuǎn)義內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IntelliJ IDEA全局內(nèi)容搜索和替換教程圖解
很多朋友在做項目時,會在整個項目里活指定文件夾下進(jìn)行全局搜索和替換,下面小編給大家?guī)砹薎ntelliJ IDEA全局內(nèi)容搜索和替換教程圖解,需要的朋友參考下吧2018-04-04
Java JVM字節(jié)碼指令集總結(jié)整理與介紹
本節(jié)將會著重介紹一下JVM中的指令集、Java是如何跨平臺的、JVM指令集參考手冊等內(nèi)容。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09
Scala數(shù)據(jù)庫連接池的簡單實現(xiàn)
本文主要介紹了Scala數(shù)據(jù)庫連接池的簡單實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02
Spring Boot學(xué)習(xí)入門之統(tǒng)一異常處理詳解
我們在做Web應(yīng)用的時候,請求處理過程中發(fā)生錯誤是非常常見的情況。下面這篇文章主要給大家介紹了關(guān)于Spring Boot學(xué)習(xí)入門之統(tǒng)一異常處理的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-09-09
SpringSecurity實現(xiàn)圖形驗證碼功能的實例代碼
Spring Security 的前身是 Acegi Security ,是 Spring 項目組中用來提供安全認(rèn)證服務(wù)的框架。這篇文章主要介紹了SpringSecurity實現(xiàn)圖形驗證碼功能,需要的朋友可以參考下2018-10-10
Java使用Geodesy進(jìn)行地理計算的技術(shù)指南
在地理信息系統(tǒng) (GIS) 和導(dǎo)航應(yīng)用中,精確的地理計算是基礎(chǔ),Geodesy 是一個流行的 Java 庫,用于處理地理位置、距離、方向等相關(guān)計算,本博客將介紹 Geodesy 的核心功能,并提供詳細(xì)的實踐樣例,幫助開發(fā)者快速上手,需要的朋友可以參考下2025-02-02

