MyBatis Plus 如何实现动态排序@杨章隐

本文介绍了如何在MyBatisPlus的强大功能基础上,解决前端多字段排序的挑战,通过前端传参处理驼峰命名转下划线,并提供了一个安全的排序类和工具类示例,确保了代码的可维护性和安全性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

目录

1、应用场景

2、具体思路

3、具体实现

        a、排序类

        b、工具类(主要驼峰转蛇形,如果项目已经有工具,则修改上面代码)

        c、需要排序的VO直接继承Sorter,如以下代码


1、应用场景

MyBatis Plus功能很强大,但是对于前端多个排序字段并存且自由度高的场景下,服务端人员很难通过原来的MyBatis Plus功能进行编码,所以有了这个文章。

2、具体思路

通过前端传过来的参数对参数进行转驼峰拼接成order Sql 传入排序字段

(存在注入风险,自己做好防护)

(存在注入风险,自己做好防护)

(存在注入风险,自己做好防护)

3、具体实现

        a、排序类


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 (StringUtil.isNotEmpty(sortColumn)) {
            // 驼峰命名转为下划线
            sort = StringUtil.toUnderScoreCase(sortColumn);

            if (sort.contains(",")) {
                sortArray = sort.split(",");
            }
        } else {
            return "";
        }
        if (StringUtil.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 (StringUtil.isNotEmpty(statement)) {
            return " order by " + statement;
        } else {
            return statement;
        }
    }
}

        b、工具类(主要驼峰转蛇形,如果项目已经有工具,则修改上面代码)


/**
 * 3. 字符串工具类
 * 4.  5. @author lcl
 */
public class StringUtil 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();
    }
}

        c、需要排序的VO直接继承Sorter,如以下代码


@Data
@ApiModel("医生搜索参数对象")
public class SearchDoctorParams extends Sorter{

    @ApiModelProperty("搜索关键字")
    private String queryKeyword;
    @ApiModelProperty("科室编号")
    private String departmentNo;
    @ApiModelProperty("医生名称")
    private String doctorName;
    @ApiModelProperty("地区编码")
    private String areaCode;
    @ApiModelProperty("职称编码")
    private String positionCategory;
    @ApiModelProperty("是否有医保编码医生")
    private Boolean withInsuranceCode;
    @ApiModelProperty(value = "三甲名医  1-是 2-否")
    private Integer isFamousDoctor;
    @ApiModelProperty(value = "医生类型(10-全职医生,20-兼职医生)")
    private Integer doctorType;

    @ApiModelProperty(value = "医生ids", hidden = true)
    private List<String> doctorIds;


    @ApiModelProperty(value = "医生特长标签")
    @DisplayDictionary(category = "doctor_specialty_tag")
    private String doctorTag;

    String orderStr;
}

//给orderStr赋值
record.setOrderStr(Sorter.getOrderByStatement(record));
//xml层使用
<choose>
    <when test="params.orderStr != null and params.orderStr != ''">
        ${orderStr}
    </when>
    <otherwise>
        默认排序
    </otherwise>
</choose>
//QueryWrapper、LambdaQueryWrapper等直接在最后用last


basicDataService.getList(new LambdaQueryWrapper<>().last(Sorter.getOrderByStatement(record)));

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

杨章隐

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值