CONCAT函数使用中出现空指针异常问题分析

文章目录

前言

在项目中,遇到一个场景,使用mysql的CONCAT函数实现字段值拼接,出现如下报错:

总的来说:空指针异常。

  • 原始sql:
sql 复制代码
<select id="findFailedTaskNames" resultType="java.lang.String">
      SELECT
          CONCAT(software_name, '_', software_version, '_', developer) as taskName
      FROM
          xxx
      WHERE
          id IN
      <foreach collection="list" item="item" open="(" close=")" separator=",">
          #{item}
      </foreach>
</select>
  • 报错:

    2025-11-03 11:10:16.596 [jyh-detect-whl] traceId[] [http-nio-10002-exec-3] ERROR c.q.w.t.c.s.h.ControllerExceptionAdvice - [handleRuntimeException,83] - 请求地址'/detect/task/storage/delete/tasks',发生未知异常.
    java.lang.NullPointerException: null
    at cn.qbs.wa.teach.course.standard.service.impl.CodeCheckTaskServiceImpl.lambdadeleteFailedTasks5(CodeCheckTaskServiceImpl.java:417)
    at java.base/java.util.stream.ReferencePipeline31.accept(ReferencePipeline.java:195)
    at java.base/java.util.ArrayListArrayListSpliterator.forEachRemaining(ArrayList.java:1655) at java.base/java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:484) at java.base/java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:474) at java.base/java.util.stream.ReduceOpsReduceOp.evaluateSequential(ReduceOps.java:913)
    at java.base/java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:234)
    at java.base/java.util.stream.ReferencePipeline.collect(ReferencePipeline.java:578)
    at cn.qbs.wa.teach.course.standard.service.impl.CodeCheckTaskServiceImpl.deleteFailedTasks(CodeCheckTaskServiceImpl.java:418)
    at cn.qbs.wa.teach.course.standard.service.impl.CodeCheckTaskServiceImpl$$FastClassBySpringCGLIB$$372942fe.invoke(<generated>)
    at org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:218)
    at org.springframework.aop.framework.CglibAopProxy.invokeMethod(CglibAopProxy.java:386)
    at org.springframework.aop.framework.CglibAopProxy.access000(CglibAopProxy.java:85) at org.springframework.aop.framework.CglibAopProxyDynamicAdvisedInterceptor.intercept(CglibAopProxy.java:704)
    at cn.qbs.wa.teach.course.standard.service.impl.CodeCheckTaskServiceImpl$$EnhancerBySpringCGLIB$$e6d08e42.deleteFailedTasks(<generated>)
    at cn.qbs.wa.teach.course.standard.controller.web.TrustApplyController.deleteFailedTasks(TrustApplyController.java:152)
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.base/java.lang.reflect.Method.invoke(Method.java:566)
    at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:205)
    at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:150)
    at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:117)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:895)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:808)
    at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)
    at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1067)
    at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:963)
    at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)
    at org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:909)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:681)
    at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:764)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:227)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)
    at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)
    at org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:100)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:117)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)
    at org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:93)
    at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:117)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)

为什么?

这是因为,部分字段的值为null,在大多数数据库中(MySQL、PostgreSQL等),CONCAT 函数中任何参数为 NULL 都会导致整个结果为 NULL。

如何解决?

  • 使用 CONCAT_WS,CONCAT_WS 会忽略 NULL 值,只连接非 NULL 的部分。
  • 使用数据库特定的函数:
    • MySQL: IFNULL(字段名, '') 或 COALESCE(字段名, '')
    • PostgreSQL: COALESCE(字段名, '')
  • 使用 CASE WHEN
    • CASE WHEN 字段名 IS NULL THEN '' ELSE 字段名 END

总结

以上为个人学习分享,如有问题,欢迎指出:)

相关推荐
云老大TG:@yunlaoda3602 分钟前
如何使用华为云国际站代理商的BRS进行数据安全保障?
大数据·数据库·华为云·云计算
工具人55556 分钟前
strip()方法可以删除字符串中间空格吗
数据库·mysql
松涛和鸣10 分钟前
35、Linux IPC进阶:信号与System V共享内存
linux·运维·服务器·数据库·算法·list
xinyu_Jina18 分钟前
局域网文件传输:P2P应用层协议——元数据握手与数据通道的生命周期管理
数据库·asp.net·p2p
彬鸿科技35 分钟前
【SDR课堂第42讲】RFSOC开发入门之开发环境搭建(三)
linux·运维·数据库·ubuntu·postgresql·软件无线电·软无
九章-36 分钟前
金仓数据库助力中国石油安全环保技术研究院安全生产智能管控系统全面实现数据库国产化替代
数据库·安全
陌路2037 分钟前
redis 发布订阅功能
数据库·redis·缓存
丁丁丁梦涛39 分钟前
navicat跨服务器连接MySQL数据库
服务器·数据库·mysql
tgethe42 分钟前
mysql-索引详解
数据库·mysql
一个天蝎座 白勺 程序猿44 分钟前
Apache IoTDB(11):分段聚合深度解析——从原理到实战的完整指南
数据库·apache·iotdb