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)行了擴展,目前可以支持自定義類參數(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-04Java JVM字節(jié)碼指令集總結(jié)整理與介紹
本節(jié)將會著重介紹一下JVM中的指令集、Java是如何跨平臺的、JVM指令集參考手冊等內(nèi)容。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09Scala數(shù)據(jù)庫連接池的簡單實現(xiàn)
本文主要介紹了Scala數(shù)據(jù)庫連接池的簡單實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02Spring Boot學(xué)習(xí)入門之統(tǒng)一異常處理詳解
我們在做Web應(yīng)用的時候,請求處理過程中發(fā)生錯誤是非常常見的情況。下面這篇文章主要給大家介紹了關(guān)于Spring Boot學(xué)習(xí)入門之統(tǒng)一異常處理的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-09-09SpringSecurity實現(xiàn)圖形驗證碼功能的實例代碼
Spring Security 的前身是 Acegi Security ,是 Spring 項目組中用來提供安全認(rèn)證服務(wù)的框架。這篇文章主要介紹了SpringSecurity實現(xiàn)圖形驗證碼功能,需要的朋友可以參考下2018-10-10Java使用Geodesy進(jìn)行地理計算的技術(shù)指南
在地理信息系統(tǒng) (GIS) 和導(dǎo)航應(yīng)用中,精確的地理計算是基礎(chǔ),Geodesy 是一個流行的 Java 庫,用于處理地理位置、距離、方向等相關(guān)計算,本博客將介紹 Geodesy 的核心功能,并提供詳細(xì)的實踐樣例,幫助開發(fā)者快速上手,需要的朋友可以參考下2025-02-02