一次全表审核“卡99%”故障排查:空字符串在while循环中引发的性能陷阱

今天下午听到部门内另外一个报表项目说他们全表审核 操作前端一直卡在99%,就好奇看了一下。前端页面触发全表审核 请求,过了五分钟没有响应请求失败,看后台还继续在运行。

排查问题

全表审核 是后台是异步执行,所以直接查看线程栈,IDEA里面直接点击Get Thread Dump获取线程栈信息(也可以通过jvm命令查看,visualvm也行)。看了一下全表审核确实还没执行完成,线程一直在waiting,因为代码里面是调用了join方法,会等待所有任务执行完成拿到结果。

join卡在这,说明还有执行代码的子线程,查看线程池里面的子线程列表,发现还有一个子线程在跑,根据线程栈信息跳转到对应代码行。

这个代码在平台写的jar里面。点开发现了不对劲的地方,count的值9.9个亿,心想应该是死循环了。他这个方法看注释是【统计s字符串在text字符串中出现的次数】

仔细看这个函数,问题就出在while (true),且s字符串为空字符。

  1. 入参字符串s值为""空字符,text.indexOf(s)返回值为0
  2. 进入else条件,count用来记录出现次数的值自增+1
  3. 调用被统计字符串textsubstring方法,关键点来了,i因为是0,空字符s.length()也是0。那这个方法调用就等同于text.substring(0)substring方法形参如果为0的话,返回值是字符串对象本身。所以每次while循环执行的结果都一模一样,除了count在不停的自增。
java 复制代码
    private int count(String text, String s) {
        int count = 0;
        while (true) {
            
            int i = text.indexOf(s);
            if (i == -1) {
                break;
            } else {
                count++;
                text = text.substring(i+s.length());
            }
        }
        return count;
    }

解决方案

使用Spring框架中的StringUtil.countOccurrencesOf方法。

相关推荐
逍遥德14 分钟前
Java 锁(线程间)和数据库锁(事务间)对比详解
java·数据库·sql·高并发·锁机制
gwjcloud27 分钟前
Docker详解
java·docker·容器
河阿里36 分钟前
Java-JWT令牌技术深度指南
java·开发语言
WiChP1 小时前
【V0.1B6】从零开始的2D游戏引擎开发之路
java·log4j·游戏引擎
leaves falling1 小时前
C/C++ 的内存管理,函数栈帧详讲
java·c语言·c++
文静小土豆1 小时前
Java 应用上 K8s 全指南:从部署到治理的生产级实践
java·开发语言·kubernetes
zhimingwen1 小时前
初探 Java 後端開發:解決 macOS 環境下 Spring Boot 項目啟動的各類「坑」
java·spring boot
Rsun045512 小时前
3、Java 工厂方法模式从入门到实战
java·开发语言·工厂方法模式
田梓燊2 小时前
leetcode 142
android·java·leetcode
亚空间仓鼠2 小时前
Ansible之Playbook(三):变量应用
java·前端·ansible