当前位置: 首页 > news >正文

SpringTask 引起的错误

SpringTask 引起的错误

1. 场景

在使用 SpringBoot 编写后台程序时,当在浏览器页面中发起请求时,MP 自动填充来完成一些字段的填充,例如创建时间、创建人、更新时间、更新人等。但是当编写微信小程序时,由于一些字段无法进行自动填充,例如创建人、更新人等,这时需要进行处理,来区分是否为微信端的请求

private final HttpServletRequest request;

// 是否排除路径,排除微信端的路径,不自动填充用户id
private boolean isExclude() {
  String path = request.getRequestURI();
  return !path.startsWith("/member");
}

// 插入数据时自动填充
@Override
public void insertFill(MetaObject metaObject) {
  if (isExclude()) {
    this.strictInsertFill(metaObject, "createBy", String.class, String.valueOf(getLoginUserId()));
  }
  this.strictInsertFill(metaObject, "createTime", Date.class, DateUtils.getNowDate());
}

// 修改数据时自动填充
@Override
public void updateFill(MetaObject metaObject) {
  if (isExclude()) {
    this.setFieldValByName("updateBy", String.valueOf(getLoginUserId()), metaObject);
  }
  this.setFieldValByName("updateTime", DateUtils.getNowDate(), metaObject);
}

但是此时,就会产生一个问题:当在浏览器端(后台管理端)使用定时任务时(这里以 SpringTask 举例),就会出现异常


提示 “当前请求并未为 Web 请求”,后来经过层层排查,发现是自动填充处使用了 HttpServletRequest 来获取当前请求详情。

在场景中,isExclude 方法依赖于 HttpServletRequest 对象来获取请求路径。然而,定时任务并不属于 Web 请求的一部分,因此在定时任务执行时,HttpServletRequest 对象不可用,这会导致遇到的 IllegalStateException 异常。

2. 解决方案

为了解决这个问题,可以考虑以下几种方法:

  1. 检查请求是否存在:在 isExclude 方法中添加一个检查,判断 HttpServletRequest 是否为空,如果为空则直接返回 truefalse,具体根据你的业务需求来决定。但这只是避免了异常,并没有真正解决问题,因为定时任务确实不需要根据请求路径来判断是否填充用户信息。
  2. 使用不同的条件来判断是否排除:如果定时任务和微信端请求可以通过其他方式区分开来(比如特定的线程池、特定的任务标识等),你可以考虑使用这些方式来判断,而不是依赖请求路径。
  3. 将用户ID作为参数传递:对于定时任务,你可以在任务触发时直接传递用户ID,而不是在任务内部去获取。这样可以避免对 HttpServletRequest 的依赖。
  4. 使用不同的 MetaObjectHandler 实现:为定时任务创建一个独立的 MetaObjectHandler 实现,这个实现不需要考虑 HttpServletRequest,也不需要调用 isExclude 方法。
  5. 修改定时任务逻辑:在定时任务中,手动填充创建人或更新人字段,而不是依赖 MetaObjectHandler 来自动填充。这样可以确保定时任务在执行时不会调用 isExclude 方法。
  6. 基于任务类型进行判断:在 MetaObjectHandler 类中增加一个字段,用于标识当前操作的类型(比如是定时任务还是 Web 请求),在 insertFillupdateFill 方法中根据这个字段来决定是否执行 isExclude 方法的判断。

3. 修改

这里使用一种很简单的方法,使用 RequestContextHolder.getRequestAttributes() 方法来判断当前线程是否绑定了一个 Web 请求的上下文,这是一种常见的方法来区分 Web 请求和其他非 Web 请求(例如定时任务)的线程上下文。这种方法在 Spring 框架中被广泛用于获取当前请求的相关信息,而不会抛出 No thread-bound request found 这样的异常。

具体来说,在的 isExclude 方法中,通过检查 RequestContextHolder.getRequestAttributes() 返回的结果是否为 null,可以判断当前线程是否在一个 Web 请求的上下文中。如果返回的是 null,说明当前线程不是在一个 Web 请求中,因此可以决定不执行某些依赖于 Web 请求的操作,比如填充 createByupdateBy 字段。

private boolean isExclude() {
  // 校验是否为 Web 请求,避免非 Web 请求导致 IllegalStateException 异常
  ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
  if (attributes != null) {
    HttpServletRequest request = attributes.getRequest();
    String path = request.getRequestURI();
    return !path.startsWith("/member");
  }
  return false;
}

以下是一些关键点,帮助更好地理解这种方法:

  1. RequestContextHolder: 这个类提供了对当前请求的访问。通过 getRequestAttributes() 方法,可以获取到当前线程的请求属性。
  2. ServletRequestAttributes: 这是 RequestContextHolder.getRequestAttributes() 返回的对象类型之一,包含了当前请求的相关信息。
  3. 判断是否为 web 请求: 通过检查 RequestContextHolder.getRequestAttributes() 是否返回 null,可以判断当前线程是否在一个 web 请求的上下文中。如果不是 null,则可以安全地获取 HttpServletRequest 并进行相关操作。
  4. ThreadLocal 使用: 尽管在 MyMetaObjectHandler 中没有直接使用 ThreadLocal 来判断是否为 web 请求,但 Spring 框架本身在处理 web 请求时会使用 ThreadLocal 来存储请求的上下文信息。因此,RequestContextHolder.getRequestAttributes() 能够正确地识别当前线程是否为 web 请求线程。

这种方法不仅能够有效地区分 Web 请求和其他请求,还能避免出现 No thread-bound request found 的异常,是一个比较优雅的解决方案。如果有其他特定的需求或场景,也可以考虑自定义 ThreadLocal 变量来标识不同的请求类型,但通常情况下,使用 Spring 提供的 RequestContextHolder 就可以满足大多数需求。

相关文章:

  • 【折线图 Line】——12
  • PHP之常量
  • [数据结构]设计循环队列
  • 【由技及道】量子构建交响曲:Jenkinsfile流水线的十一维编程艺术【人工智障AI2077的开发日志008】
  • SpringMvc与Struts2
  • 【Linux】命名管道
  • 影刀RPA开发拓展--正则表达式
  • transformer架构解析{模型基本测试}(含代码)-9
  • 软件测试(三)——Bug篇
  • 002.words and phrases
  • 通过多线程获取RV1126的AAC码流
  • CVE-2025-0392:JeeWMS graphReportController.do接口SQL注入漏洞复现
  • 磁盘空间用尽导致的系统500错误(failed to openstream:No space left on device)
  • Android14 OTA差分包升级报kPayloadTimestampError (51)
  • 使用 Deepseek + kimi 快速生成PPT
  • 通过计费集成和警报监控 Elasticsearch Service 成本
  • 宇树科技再落一子!天羿科技落地深圳,加速机器人创世纪
  • HDFS 为什么不适合处理小文件?
  • PMP项目管理—沟通管理篇—补充内容
  • Java常用正则表达式(身份证号、邮箱、手机号)格式校验
  • css div网站/加速游戏流畅的软件
  • wordpress新建页面连接无法访问/seo公司排名
  • WordPress问答插件路由/排名优化seo
  • 视觉中国网站/2022年最火的新闻摘抄
  • 昌邑网站建设公司/百度如何搜索网址
  • 淘宝客网站可以做百度推广/谷歌浏览器中文手机版