MyBatis-Plus 实现动态字段排序

Posted 写一行代码二次元老婆全白给了

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MyBatis-Plus 实现动态字段排序相关的知识,希望对你有一定的参考价值。

场景

虽然前端能根据返回的数组进行对字段的排序,但也仅局限于实现当前页的排序,无法满足全部数据的排序,所以需要走接口的查询进行排序,获取最全的排序数据。

业务需求

思路

  1. 前端需传递排序的字段,该字段是正序还是倒叙。可能存在多个,所以字段为字符串数组,可以通过逗号分开。
  2. 后端拿到参数后,如果字段名是驼峰命名,需要转为小写字母+下划线命名。
  3. 将处理后的字段进行排序sql上的拼接处理后,拼接到 order by 语句后面,得到order by语句。
  4. 根据MyBatis-Plus的 last 方法,将处理的order by语句进行拼接到查询sql语句后面。

实现

  1. 前端传递的参数格式为:

  "sort": "userName,updateTime",
  "order": "asc,desc"

  1. 后端接收排序对象 Sorter ,那个请求查询用到自定义排序就继承该类:
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;

/**
 1. 基础排序对象,包含排序字段和排序方式
 */
@Data
public class Sorter


    @ApiModelProperty(value = "排序字段",example = "userName")
    private String sort;

    @ApiModelProperty(value = "排序方式",example = "asc/desc")
    private String order;

    /**
     * 根据查询条件拼接得到order by语句
     * @param sorter 分页查询条件
     * @return String
     */
    public static String getStatement(Sorter sorter)
    
        String sort;
        String[] sortArray = ;
        String[] orderArray = ;
        String order = sorter.getOrder();
        String sortColumn = sorter.getSort();
        StringBuilder statement = new StringBuilder();

        // 多字段排序
        if (StringUtils.isNotEmpty(sortColumn))
        
            // 驼峰命名转为下划线
            sort = StringUtils.toUnderScoreCase(sortColumn);

            if (sort.contains(",")) 
                sortArray = sort.split(",");
            
        
        else
        
            return "";
        
        if (StringUtils.isNotEmpty(order))
        
            if (order.contains(",")) 
                orderArray = order.split(",");
            
        
        else
        
            return "";
        

        if (sortArray.length > 0 && orderArray.length > 0)
        
            int length = sortArray.length;

            for (int i = 0; i < length; i++) 
                statement.append(sortArray[i]);
                statement.append(" ");
                statement.append(orderArray[i]);

                if (i < length - 1 ) 
                    statement.append(", ");
                
            
        
        else
        
            // " #sort #order“
            statement.append(sort);
            statement.append(" ");
            statement.append(order);
        
        return statement.toString();
    

    /**
     * 根据查询条件拼接得到order by语句
     * @param sorter 分页查询条件
     * @return String
     */
    public static String getOrderByStatement(Sorter sorter)
    
        String statement = getStatement(sorter);

        if (StringUtils.isNotEmpty(statement))
        
            return " order by " + statement;
        
        else
        
            return statement;
        
    

  1. 处理字段驼峰式的工具类:
/**
 3. 字符串工具类
 4.  5. @author lcl
 */
public class StringUtils extends org.apache.commons.lang3.StringUtils

    /** 下划线 */
    private static final char SEPARATOR = \'_\';

    /**
     * * 判断一个字符串是否为非空串
     *
     * @param str String
     * @return true:非空串 false:空串
     */
    public static boolean isNotEmpty(String str)
    
        return !isEmpty(str);
    

    /**
     * 驼峰转下划线命名
     */
    public static String toUnderScoreCase(String str)
    
        if (str == null)
        
            return null;
        
        StringBuilder sb = new StringBuilder();
        // 前置字符是否大写
        boolean preCharIsUpperCase = true;
        // 当前字符是否大写
        boolean curreCharIsUpperCase = true;
        // 下一字符是否大写
        boolean nexteCharIsUpperCase = true;
        for (int i = 0; i < str.length(); i++)
        
            char c = str.charAt(i);
            if (i > 0)
            
                preCharIsUpperCase = Character.isUpperCase(str.charAt(i - 1));
            
            else
            
                preCharIsUpperCase = false;
            

            curreCharIsUpperCase = Character.isUpperCase(c);

            if (i < (str.length() - 1))
            
                nexteCharIsUpperCase = Character.isUpperCase(str.charAt(i + 1));
            

            if (preCharIsUpperCase && curreCharIsUpperCase && !nexteCharIsUpperCase)
            
                sb.append(SEPARATOR);
            
            else if ((i != 0 && !preCharIsUpperCase) && curreCharIsUpperCase)
            
                sb.append(SEPARATOR);
            
            sb.append(Character.toLowerCase(c));
        
        return sb.toString();
    

  1. 请求查询的入参对象继承 Sorter 排序对象:

入参对象
  1. Mapper层直接调用last在sql尾拼接语句方法,将SortergetOrderByStatement(Sorter sorter)方法传入:

语句拼接

结果

测试排序

以上是关于MyBatis-Plus 实现动态字段排序的主要内容,如果未能解决你的问题,请参考以下文章

MyBatis-Plus 条件构造器(动态字段名)

MyBatis-Plus实现动态表名

Mybatis-Plus实现公共字段自动赋值

Mybatis-plus@DS实现动态切换数据源应用

如何解决mybatis-plus调用update方法时,自动填充字段不生效问题

Mybatis-Plus:条件构造器(allEq基本比较操作模糊查询排序逻辑查询select)