Spring Boot Web 应用请求响应太慢故障排查
创始人
2024-11-15 00:33:03
0

Spring Boot Web 应用请求响应太慢排查

线上接口响应太慢,Spring Boot + 内嵌的 Tomcat 服务器,Tomcat 处理请求的最大线程数普通情况是 150 左右,最大是 200,所以当同时处理的请求过多,并且每个请求一直没有处理完成。所有的线程都在繁忙,没有办法处理新的请求,就会导致新的请求排队等待处理,从而造成了迟迟无法响应的线上事故,用户体验太差。

场景复现

一个简单的 controller 接口

@RestController public class SpringBootController {  	@GetMapping("/") 	public String index() throws InterruptedException {  		// 每个线程都睡眠,模拟业务接口响应慢的问题 		Thread.sleep(60 * 30 * 1000); 		System.out.println("current thread: " + Thread.currentThread().getName());  		return "Hello Spring Boot!"; 	}  } 

随后设置 tomcat 服务器的最大线程为 5 :

server:   port: 8080   tomcat:     threads:       max: 5  spring:   application:     name: simple-application      

测试

之后我们在单元测试中模拟 10 个线程对接口发起请求

public class RequestTest {  	public static void main(String[] args) { 		 		for (int i = 0; i < 10; i++) { 			new Thread(new RequestTask()).start(); 		} 	}  	private static class RequestTask implements Runnable { 		@Override 		public void run() { 			try { 				URL url = new URL("http://localhost:8080/"); 				HttpURLConnection connection = (HttpURLConnection) url.openConnection(); 				connection.setRequestMethod("GET"); 				int responseCode = connection.getResponseCode(); 				System.out.println("Response Code: " + responseCode); 			} 			catch (IOException e) { 				e.printStackTrace(); 			} 		} 	} 	 } 

首先,我们注释掉 controller 中的睡眠代码,运行测试得到以下的输出:

spring boot web 应用响应输出: 2024-08-01T15:56:25.369+08:00  INFO 7304 --- [simple-application] [nio-8080-exec-2] o.s.web.servlet.DispatcherServlet        : Completed initialization in 0 ms current thread: http-nio-8080-exec-4 current thread: http-nio-8080-exec-5 current thread: http-nio-8080-exec-1 current thread: http-nio-8080-exec-2 current thread: http-nio-8080-exec-3 current thread: http-nio-8080-exec-5 current thread: http-nio-8080-exec-2 current thread: http-nio-8080-exec-4 current thread: http-nio-8080-exec-1 current thread: http-nio-8080-exec-3      测试输出: Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 Response Code: 200 

我们看到线程在响应输出。接下来,打开注释代码,模拟事故现场。将会在控制台看不到任何输出,测试方法也会卡住不动。

排查

我们使用 JVM 的一些命令来进行排查:

# 首先查看 java 的进程 id C:\Users\Administrator>jps -l 18400 indi.yuluo.tomcat.SpringBootApplication 10324 jdk.jcmd/sun.tools.jps.Jps 16872 org.jetbrains.jps.cmdline.Launcher 18072 org.jetbrains.idea.maven.server.RemoteMavenServer36 9624 7372 inid.yuluo.tomcat.RequestTest 

我们看到 SpringBootApplication 和 RequestTest 在运行,之后使用 jstack 命令生成线程快照,并保存为文件。

C:\Users\Administrator>jstack 18400 > C:\Users\Administrator\Desktop\spring-boot-thread_dump.txt 

打开线程快照文件,搜索 http-nio 就能看到 Tomcat 的请求处理线程,所有的请求处理线程状态都是 TIMED_WAITING ,表示线程正在等待另一个线程执行特定的动作,但是有一个指定的等待时间。而且能直接看到请求是阻塞在了哪个代码位置。

"http-nio-8080-exec-2" #28 daemon prio=5 os_prio=0 cpu=0.00ms elapsed=187.14s tid=0x0000015ffe6089f0 nid=0x1ee4 waiting on condition  [0x00000026eb5fc000] 						# `TIMED_WAITING`    java.lang.Thread.State: TIMED_WAITING (sleeping) 	at java.lang.Thread.sleep(java.base@17.0.1/Native Method) 	# 阻塞位置: 	at indi.yuluo.tomcat.controller.SpringBootController.index(SpringBootController.java:18) 

解决方案

适当修改 Tomcat 的最大线程数,可以增加并发请求的处理能力。

适当调大 Tomcat 的最小空闲线程数,可以确保在并发高峰时刻,Tomcat 能迅速响应新的请求,而不需要重新创建线程。

修改值需要对用户体量做出预估之后,进行测试之后确定。

相关内容

热门资讯

一分钟内幕!科乐吉林麻将系统发... 一分钟内幕!科乐吉林麻将系统发牌规律,福建大玩家确实真的是有挂,技巧教程(有挂ai代打);所有人都在...
一分钟揭秘!微扑克辅助软件(透... 一分钟揭秘!微扑克辅助软件(透视辅助)确实是有挂(2024已更新)(哔哩哔哩);1、用户打开应用后不...
五分钟发现!广东雀神麻雀怎么赢... 五分钟发现!广东雀神麻雀怎么赢,朋朋棋牌都是是真的有挂,高科技教程(有挂方法)1、广东雀神麻雀怎么赢...
每日必看!人皇大厅吗(透明挂)... 每日必看!人皇大厅吗(透明挂)好像存在有挂(2026已更新)(哔哩哔哩);人皇大厅吗辅助器中分为三种...
重大科普!新华棋牌有挂吗(透视... 重大科普!新华棋牌有挂吗(透视)一直是有挂(2021已更新)(哔哩哔哩)1、完成新华棋牌有挂吗的残局...
二分钟内幕!微信小程序途游辅助... 二分钟内幕!微信小程序途游辅助器,掌中乐游戏中心其实存在有挂,微扑克教程(有挂规律)二分钟内幕!微信...
科技揭秘!jj斗地主系统控牌吗... 科技揭秘!jj斗地主系统控牌吗(透视)本来真的是有挂(2025已更新)(哔哩哔哩)1、科技揭秘!jj...
1分钟普及!哈灵麻将攻略小,微... 1分钟普及!哈灵麻将攻略小,微信小程序十三张好像存在有挂,规律教程(有挂技巧)哈灵麻将攻略小是一种具...
9分钟教程!科乐麻将有挂吗,传... 9分钟教程!科乐麻将有挂吗,传送屋高防版辅助(总是存在有挂)1、完成传送屋高防版辅助透视辅助安装,帮...
每日必看教程!兴动游戏辅助器下... 每日必看教程!兴动游戏辅助器下载(辅助)真是真的有挂(2025已更新)(哔哩哔哩)1、打开软件启动之...