1用户需求

使用达梦数据库(Mysql同理),查询结果,按照某些字段进行排序,将为null的值放到最后。按照更新时间排序,但是更新时间可能为null,因此将null的数据放到最后。

2解决方案

最简单的方式,当然是下面这种直接在SQL最后面 NULLS LAST ,但是问题是,我都用MybatisPlus,下面的这种SQL那肯定不会写了啊,要是用MybatisPlus还写下面这种单表SQL的查询的,我建议可以放弃MybatisPlus了

--达梦数据库
SELECT * FROM users ORDER BY OPERATE_DATE ASC NULLS LAST 
-- mysql
SELECT * FROM users ORDER BY OPERATE_DATE IS NULL, OPERATE_DATE ASC

先说最终解决方案,用mybatis拦截器修改最终执行的sql语句

达梦数据库下如何解决嗯?
思路就是将queryWrapper构造的SQL语句中的ASC替换成ASC NULLS LAST
即使用queryWrapper的orderBy时,mybatis-plus会生成这个SQL语句
SELECT * FROM users ORDER BY OPERATE_DATE ASC
而我们要做的就是在mybatis-plus执行之前,将ASC变成 ASC NULLS LAST 

下面是我们进行排序的代码。目前来看,我们只能改这里,不过查找了一圈,都没有解决方案,因此放弃,用另外拦截器的方式实现。

if(!ObjectUtils.isEmpty(orderBy)) {
	if(orderBy instanceof Collection) {
		String[] array = ((Collection<?>) orderBy).toArray(new String[0]);
		queryWrapper.orderBy(true, isAsc, Arrays.asList(array));
	}else {
		queryWrapper.orderBy(true, isAsc, orderBy.toString());

当然GPT一本正经的胡说八道,看着挺像回事的,可惜mybait-plus没有这个方法,所以看看就好。

orderByAscWithNullsLast()

3拦截器代码

这里开始,就是最后的代码实现了

3.1编写拦截器LastNullInterceptor

import java.lang.reflect.Field;
import java.sql.Connection;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.SqlCommandType;
import org.apache.ibatis.mapping.StatementType;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.SystemMetaObject;
import org.springframework.util.ReflectionUtils;
import com.baomidou.mybatisplus.core.toolkit.PluginUtils;
 * @description:拦截查询SQL,处理查询SQL中的排序
 * @author:hutao
 * @mail:[email protected]
 * @date:2023年7月25日 下午12:17:50
@Intercepts(@Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class}))
public class LastNullInterceptor implements Interceptor {
	//,有兴趣,可以看看MybatisPlusInterceptor怎么实现的
    private static final String DESC = "DESC";
    private static final String ASC = "ASC";
    private static final String REPLACE_DESC = "DESC NULLS LAST";
    private static final String REPLACE_ASC = "ASC NULLS LAST";
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
    	StatementHandler handler = PluginUtils.realTarget(invocation.getTarget());
        MetaObject metaObject = SystemMetaObject.forObject(handler);
        // 判断是不是SELECT操作,跳过存储过程
        MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement");
        if (SqlCommandType.SELECT != mappedStatement.getSqlCommandType()
                || StatementType.CALLABLE == mappedStatement.getStatementType()) {
            return invocation.proceed();
        BoundSql boundSql = handler.getBoundSql();
        String sql = boundSql.getSql().toUpperCase();
        if(sql.contains("ORDER BY")) {
        	sql = this.replaceLast(sql, DESC, REPLACE_DESC);
        	sql = this.replaceLast(sql, ASC, REPLACE_ASC);
        	Field sqlField = boundSql.getClass().getDeclaredField("sql");
        	ReflectionUtils.makeAccessible(sqlField);
        	ReflectionUtils.setField(sqlField, boundSql, sql);
        return invocation.proceed();
     * @description:替换最后一个字符串
     * @author:hutao
     * @mail:[email protected]
     * @date:2023年7月25日 下午2:22:21
    public String replaceLast(String str, String target, String replacement) {
    	if (str == null || target == null || replacement == null) {
            return str;
        int lastIndex = str.lastIndexOf(target);
        if (lastIndex < 0) {
            return str;
        return str.substring(0, lastIndex) + replacement + str.substring(lastIndex + target.length());

3.2注入拦截器

@SpringBootConfiguration
public class MybatisConfig {
	@Bean
    public LastNullInterceptor nullsLastInterceptor() {
        return new LastNullInterceptor();

4结果展示

打印mybatis-plus的sql,我们可以发现,已经将ASC替换成 ASC NULLS LAST了
在这里插入图片描述

备注:评论区有小伙伴说,使用

queryWrapper.last(" nulls last ")

这种方式是在sql的最后拼接上,所以,如果有类似分页的这种,还需要在最后加上limit 等关键字的就会报错。

但是问题是,我都用MybatisPlus,下面的这种SQL那肯定不会写了啊,要是用MybatisPlus还写下面这种单表SQL的查询的,我建议可以放弃MybatisPlus了。查询结果,按照某些字段进行排序,将为null的值放到最后。按照更新时间排序,但是更新时间可能为null,因此将null的数据放到最后。目前来看,我们只能改这里,不过查找了一圈,都没有解决方案,因此放弃,用另外拦截器的方式实现。当然GPT一本正经的胡说八道,看着挺像回事的,可惜mybait-plus没有这个方法,所以看看就好。 springboot + mybatisplus + holo 需求是NULL在最末尾,用QueryWrapper查询排序在最前面。 这里有两个处理方法: 1、在xml里面写 DESC NULLS LAST 2、用拦截器统一处理。 mybatis拦截器排序统一增加nulls last mybatis拦截器代码: package com.dwanins.demo.interceptor; import com.baomidou.mybatisplus.core.toolkit.Pl
在MySQL中,NULL意味着未知NULL不是零或空字符”NULL不等于其自身。如果将NULL与另一个NULL或任何其他进行比较,则结果为NULL,因为每个NULL都是未知的。通常,使用NULL来表示数据丢失,未知或不适用。例如,当客户的电话号码可设置为NULL,并且可以稍后添加。当创建表,可以通过使用NOTNULL约束来指定列是否接受NULL。例如,以下语句创建leads表:因此,id是主键列,因此它不接受任何NULL。customer_name和source列使用NOTNULL约束,因此,不能在这些列中插入NULL。email和phone列是可以接受N
MybatisPlus的selectOne查询如果没有符合条件的数据,则返回null。selectOne查询的作用是查询数据库中符合条件的一条数据,如果有多条符合条件的数据,则只返回第一条数据。如果没有符合条件的数据,则返回null。在使用selectOne查询,需要注意是否会返回null,以便进行后续的判断和处理。同,也需要注意使用selectOne查询的条件是否准确,保证查询结果的正确性。如果需要查询多条数据,则应该使用selectList查询MybatisPlus是一个强大的ORM框架,使用简单,支持多种数据库,可以大大提高Java开发效率。