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

总结

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

相关推荐
一 乐1 小时前
婚纱摄影网站|基于ssm + vue婚纱摄影网站系统(源码+数据库+文档)
前端·javascript·数据库·vue.js·spring boot·后端
1.14(java)2 小时前
SQL数据库操作:从CRUD到高级查询
数据库
Full Stack Developme3 小时前
数据库索引的原理及类型和应用场景
数据库
IDC02_FEIYA4 小时前
SQL Server 2025数据库安装图文教程(附SQL Server2025数据库下载安装包)
数据库·windows
辞砚技术录5 小时前
MySQL面试题——联合索引
数据库·面试
萧曵 丶5 小时前
MySQL 主键不推荐使用 UUID 的深层原因
数据库·mysql·索引
小北方城市网5 小时前
分布式锁实战指南:从选型到落地,避开 90% 的坑
java·数据库·redis·分布式·python·缓存
毕设十刻5 小时前
基于Vue的人事管理系统67zzz(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
前端·数据库·vue.js
TDengine (老段)7 小时前
TDengine Python 连接器入门指南
大数据·数据库·python·物联网·时序数据库·tdengine·涛思数据
萧曵 丶8 小时前
事务ACID特性详解
数据库·事务·acid