關(guān)于mybatis plus 中的查詢優(yōu)化問題
傳統(tǒng)的mybatis plus 查詢寫法
對于常規(guī)的mybatis單表查詢,我們既可以采用LambdaQueryWrapper查詢,也可以使用QueryWrapper查詢。
LambdaQueryWrapper具有防誤寫、規(guī)范代碼等好處,但是缺點是無法在復(fù)雜的多表查詢中使用。
相比較來說,使用QueryWrapper編寫查詢更加靈活,可以適應(yīng)更復(fù)雜的查詢場景。
我們首先看一個QueryWrapper查詢的例子
public List<UserMo> list (UserForm userForm) {
QueryWrapper<UserMo> queryWrapper = new QueryWrapper<>();
queryWrapper.like(StringTool.isNotEmpty(userForm.getUserName(), "name", userForm.getUserName());
queryWrapper.eq(StringTool.isNotEmpty(userForm.getMobile(), "mobile", userForm.getMobile());
// 其它的查詢條件...
return userMapper.selectList(queryWrapper);
}
對于上面的查詢語句來說,可以很好的對前端傳值進行處理,當(dāng)userForm中有前端傳值的話,就會往SQL語句中加一條where條件。
但是這樣做的話會有一個相對來說比較復(fù)雜的點,那就是當(dāng)UserForm中的字段過于多的時候,我們也許得寫十幾行的這種重復(fù)判斷的語句。
通過自定義注解來解決通用查詢條件過多問題
通過觀察mybatis plus 對于queryWrapper相關(guān)查詢方法的列子,我們可以找出一類通用方法

可以看出來這幾個方法都是傳的同樣的三個參數(shù)。
我想對于這些簡單的通用的查詢條件,也許可以有一個通用的方法來填充。
我首先設(shè)置了一個枚舉類,將這些查詢條件列出來,并在構(gòu)造方法中,將對應(yīng)的方法以反射的方式取到。
public enum QueryConditionEnum {
EQ("eq"),
NE("ne"),
GT("gt"),
GE("ge"),
LT("lt"),
LE("le"),
LIKE("like"),
NOT_LIKE("notLike"),
LIKE_LEFT("likeLeft"),
LIKE_RIGHT("likeRight");
private String name;
private Method method;
QueryConditionEnum (String name) {
this.name = name;
try {
Method method = AbstractWrapper.class.getDeclaredMethod(name, boolean.class, Object.class, Object.class);
this.method = method;
} catch (NoSuchMethodException e) {
}
}
}
再者,我想通過注解的方式來規(guī)定需要以什么方法填充,默認為EQ,對此寫了一個QueryCondition注解。
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface QueryCondition {
/**
* 默認查詢方式
*
* @return
*/
QueryConditionEnum value() default QueryConditionEnum.EQ;
/**
* 是否填充默認查詢條件
*
* @return
*/
boolean isCondition() default true;
}
然后就可以這樣構(gòu)造UserForm
public class UserForm {
private String name;
@QueryCondition(QueryConditionEnum.LIKE)
private String mobile;
}
我們需要一個工具類填充查詢條件,這里我們新增了一個參數(shù) mo對象,這是因為我們的主查詢對象是Mo對象,Mo對象存儲了相關(guān)表格名稱、表格字段名信息。
@TableName("user")
public class UserMo {
@TableField("name")
private String name;
@TableField("mobile")
private String mobile;
}
public class QueryTool {
/**
* 填充默認查詢
* @param baseClazz mo對象class
* @param queryWrapper 查詢條件
* @param form 請求對象
*/
public static void paddingDefaultConditionQuery(Class baseClazz, QueryWrapper queryWrapper, Object form) {
try {
for (Field declaredField : form.getClass().getDeclaredFields()) {
declaredField.setAccessible(true);
Object fieldValue = declaredField.get(form);
QueryCondition queryCondition = declaredField.getAnnotation(QueryCondition.class);
if (fieldValue == null) {
continue;
}
if (queryCondition == null) {
queryWrapper.eq(StringTool.isNotEmpty(fieldValue.toString()),
QueryTool.getTableName(baseClazz) + "." + QueryTool.getTableFieldName(baseClazz, declaredField),
fieldValue.toString());
continue;
}
if (queryCondition.isCondition() == false) {
continue;
}
Method method = queryCondition.value().getMethod();
method.invoke(queryWrapper, StringTool.isNotEmpty(fieldValue.toString()),
QueryTool.getTableName(baseClazz) + "." + QueryTool.getTableFieldName(baseClazz, declaredField),
fieldValue.toString());
}
} catch (Exception e) {
throw new RuntimeException("填充默認的SQL條件出錯", e);
}
}
/**
* 填充默認排序
*
* @param queryWrapper
* @param pageForm
*/
public static void paddingDefaultOrderQuery(QueryWrapper queryWrapper, PageForm pageForm) {
queryWrapper.orderBy(pageForm != null && StringTool.isNotEmpty(pageForm.getColumnName()),
pageForm.getIsAsc() == null ? false : pageForm.getIsAsc(), pageForm.getColumnName());
}
/**
* 獲取表名稱
*
* @return
*/
public static String getTableName(Class baseClazz) {
TableName tableName = (TableName) baseClazz.getDeclaredAnnotation(TableName.class);
if (tableName != null && StringTool.isNotEmpty(tableName.value())) {
return tableName.value();
}
return StringTool.toUnderline(baseClazz.getClass().getName());
}
/**
* 獲取字段名
*
* @param field
* @return
*/
public static String getTableFieldName(Class baseClazz, Field field) {
Field baseField = null;
try {
baseField = baseClazz.getDeclaredField(field.getName());
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
if (baseField == null) {
baseField = field;
}
TableId tableId = baseField.getAnnotation(TableId.class);
if (tableId != null && StringTool.isNotEmpty(tableId.value())) {
return tableId.value();
}
TableField tableField = baseField.getAnnotation(TableField.class);
if (tableField != null && StringTool.isNotEmpty(tableField.value())) {
return tableField.value();
}
return StringTool.toUnderline(baseField.getName());
}
}
最后我們就可以使用工具類來填充了 。
public List<UserMo> list (UserForm userForm) {
QueryWrapper<UserMo> queryWrapper = new QueryWrapper<>();
QueryTool.paddingDefaultConditionQuery(UserMo.class, queryWrapper, userForm);
return userMapper.selectList(queryWrapper);
}
可以看到這樣大大減少了需要填充的字段。如果有特殊字段,也能通過注解方式,跳過特殊字段,再自行填充就好。
到此這篇關(guān)于關(guān)于mybatis plus 中的查詢優(yōu)化的文章就介紹到這了,更多相關(guān)mybatis plus 查詢優(yōu)化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot restTemplate連接池整合方式
這篇文章主要介紹了springboot restTemplate連接池整合方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-10-10
java 利用反射機制,獲取實體所有屬性和方法,并對屬性賦值
這篇文章主要介紹了 java 利用反射機制,獲取實體所有屬性和方法,并對屬性賦值的相關(guān)資料,需要的朋友可以參考下2017-01-01
Java aop面向切面編程(aspectJweaver)案例詳解
這篇文章主要介紹了Java aop面向切面編程(aspectJweaver)案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08
Spring Cloud引入Eureka組件,完善服務(wù)治理
這篇文章主要介紹了Spring Cloud引入Eureka組件,完善服務(wù)治理的過程詳解,幫助大家更好的理解和使用spring cloud,感興趣的朋友可以了解下2021-02-02
Java URL自定義私有網(wǎng)絡(luò)協(xié)議
URI與URL的區(qū)別 一.先來序言一段 二.協(xié)議的自定義的理解 三.自定義協(xié)議與URL的關(guān)系 四.URL自定義私有協(xié)議實戰(zhàn) 五.后話,自定義mineType解析器2016-04-04
從內(nèi)存地址解析Java的static關(guān)鍵字的作用
這篇文章主要介紹了從內(nèi)存地址解析Java的static關(guān)鍵字的作用,包括靜態(tài)成員變量和靜態(tài)方法等重要內(nèi)容,需要的朋友可以參考下2015-10-10

