Kotlin 协程 vs RxJava vs 线程池:性能与场景对比
1. 轻量级任务:10,000 个并发延迟操作
假设需要并发执行 10,000 个非阻塞延迟任务(如模拟定时请求):
线程池实现
ExecutorService executor = Executors.newFixedThreadPool(64); // 最多 64 线程
List<Future<?>> futures = new ArrayList<>();
for (int i = 0; i < 10_000; i++) {futures.add(executor.submit(() -> {Thread.sleep(100); // 阻塞线程return null;}));
}
// 需要手动等待所有 Future 完成
- 问题:线程池最大 64 线程,处理 10,000 任务需要排队,总耗时至少
(10000/64)*100ms ≈ 15.6s
。 - 缺点:线程被阻塞,无法复用;内存占用高(每个线程约 1MB)。
RxJava 实现
Observable.range(1, 10_000).flatMap(i -> Observable.just(i).delay(100, TimeUnit.MILLISECONDS, Schedulers.io())).subscribe();
- 问题:RxJava 的
delay
内部使用线程池调度,同样受限于Schedulers.io()
的线程数(默认无上限,但实际受系统限制)。 - 缺点:频繁创建
Observable
实例和线程切换,GC 压力大。
Kotlin 协程实现
val scope = CoroutineScope(Dispatchers.Default)
scope.launch {val jobs = List(10_000) {launch { delay(100) // 非阻塞挂起,线程可被其他协程复用// 执行任务}}jobs.joinAll()
}
- 优势:协程在挂起时不占用线程,
Dispatchers.Default
默认使用 CPU 核心数线程(如 8 线程),但 10,000 个协程可高效复用这些线程,总耗时仅约100ms + 调度开销
。 - 关键点:协程的挂起(
delay
)不阻塞线程,线程利用率接近 100%。
2. 复杂异步流:链式网络请求
假设需要先请求用户信息,再根据结果并发请求订单和消息列表,最后合并结果:
线程池 + Future
CompletableFuture<User> userFuture = CompletableFuture.supplyAsync(() -> getUser(), executor);
CompletableFuture<Order> orderFuture = userFuture.thenComposeAsync(user -> getOrder(user.id), executor);
CompletableFuture<List<Message>> messagesFuture = userFuture.thenComposeAsync(user -> getMessages(user.id), executor);
CompletableFuture.allOf(orderFuture, messagesFuture).thenAcceptAsync(v -> {Order order = orderFuture.get();List<Message> messages = messagesFuture.get();showUI(order, messages);}, uiExecutor);
- 问题:回调嵌套复杂,错误处理需手动传递,且
thenComposeAsync
每次切换线程可能引入上下文开销。
RxJava 实现
getUserObservable().flatMap(user -> Observable.zip(getOrderObservable(user.id).subscribeOn(Schedulers.io()),getMessagesObservable(user.id).subscribeOn(Schedulers.io()),(order, messages) -> Pair.create(order, messages))).observeOn(AndroidSchedulers.mainThread()).subscribe(pair -> showUI(pair.first, pair.second));
- 问题:
flatMap
+zip
的嵌套导致多次线程切换,且需手动管理subscribeOn
和observeOn
。 - 缺点:内存中可能保留中间状态的
Observable
对象,增加 GC 压力。
Kotlin 协程实现
lifecycleScope.launch { // Android 生命周期绑定val user = async(Dispatchers.IO) { getUser() }.await()val (order, messages) = coroutineScope {val orderDeferred = async(Dispatchers.IO) { getOrder(user.id) }val messagesDeferred = async(Dispatchers.IO) { getMessages(user.id) }orderDeferred.await() to messagesDeferred.await()}withContext(Dispatchers.Main) { // 切回 UI 线程showUI(order, messages)}
}
- 优势:同步写法表达异步逻辑,通过
coroutineScope
自动管理子协程生命周期。 - 性能:两次
async
并发执行,复用Dispatchers.IO
线程池,无额外线程切换开销。
3. 高吞吐数据流:每秒处理 10,000 个事件
假设需要处理一个高频事件流(如传感器数据),进行过滤、转换和批量存储。
线程池 + BlockingQueue
BlockingQueue<Data> queue = new LinkedBlockingQueue<>();
// 生产者线程池
Executors.newSingleThreadExecutor().submit(() -> {while (true) {Data data = readSensor();queue.put(data); // 可能阻塞}
});
// 消费者线程池
Executors.newFixedThreadPool(8).submit(() -> {while (true) {Data data = queue.take();process(data); // 计算密集型任务}
});
- 问题:生产者-消费者模型依赖阻塞队列,线程数过多时上下文切换开销显著。
RxJava 实现
Flowable<Data> sensorFlow = Flowable.create(emitter -> {while (!emitter.isCancelled()) {Data data = readSensor();emitter.onNext(data);}
}, BackpressureStrategy.BUFFER);sensorFlow.onBackpressureBuffer().parallel(8) // 并行处理.runOn(Schedulers.computation()).map(data -> process(data)).sequential().subscribe();
- 问题:背压策略(如
BUFFER
)可能导致内存溢出;parallel
操作符的线程管理复杂。
Kotlin 协程 + Channel
val dataChannel = Channel<Data>(capacity = Channel.UNLIMITED) // 无界缓冲区// 生产者协程
launch(Dispatchers.IO) {while (true) {val data = readSensor()dataChannel.send(data)}
}// 消费者协程(启动 8 个并行处理)
repeat(8) {launch(Dispatchers.Default) { // 使用 CPU 密集型调度器for (data in dataChannel) {process(data)}}
}
- 优势:
Channel
提供非阻塞的生产者-消费者模型,Dispatchers.Default
根据 CPU 核心数优化并行度。 - 性能:协程挂起替代线程阻塞,缓冲区无锁实现,吞吐量更高。
4. 资源敏感场景:低内存设备下的并发
假设在 Android 低端设备(内存 1GB)中处理 1000 个并发 HTTP 请求:
- 线程池崩溃:创建 1000 个线程直接导致 OOM(每个线程占 1MB → 总需求 1GB)。
- RxJava 内存压力:大量
Observable
和Disposable
对象导致频繁 GC。 - 协程实现:
val requests = (1..1000).map { i ->lifecycleScope.async(Dispatchers.IO) {api.fetchData(i) // 挂起函数,复用线程} } requests.awaitAll()
- 协程内存占用仅约 1000 * 1KB = 1MB,且
Dispatchers.IO
动态调整线程池。
- 协程内存占用仅约 1000 * 1KB = 1MB,且
5. 典型场景性能对比表
场景 | 线程池 | RxJava | Kotlin 协程 |
---|---|---|---|
10,000 延迟任务 | ~15.6s (64线程) | ~15s + GC 抖动 | ~100ms |
链式网络请求 | 回调地狱,易遗漏错误 | 链式清晰,但内存占用高 | 同步写法,自动取消子任务 |
高频事件流处理 | 阻塞队列导致吞吐量瓶颈 | 背压管理复杂 | Channel 非阻塞,高吞吐 |
低内存设备并发 | OOM 崩溃 | 高频 GC | 内存占用低,无 OOM |
总结:何时选择哪种技术?
-
协程:
- 高并发 I/O 任务(网络、数据库)。
- 需要结构化并发和取消逻辑的场景。
- 资源受限环境(如移动设备)。
-
RxJava:
- 复杂事件流变换(如防抖、窗口操作)。
- 需要响应式编程范式的跨平台逻辑。
-
线程池:
- 简单后台任务,无需复杂生命周期管理。
- 依赖传统 Java 库的阻塞式 API。
进一步优化建议
- 协程调试:
添加-Dkotlinx.coroutines.debug
JVM 参数,日志中会显示协程 ID。 - RxJava 内存泄漏:
使用CompositeDisposable
集中管理订阅。 - 线程池监控:
通过ThreadPoolExecutor
的getActiveCount()
等方法实时监控负载。
通过具体场景对比可见,Kotlin 协程在资源利用、代码可读性和性能上表现更优,尤其适合现代异步编程需求。
相关文章:
Kotlin 协程 vs RxJava vs 线程池:性能与场景对比
1. 轻量级任务:10,000 个并发延迟操作 假设需要并发执行 10,000 个非阻塞延迟任务(如模拟定时请求): 线程池实现 ExecutorService executor Executors.newFixedThreadPool(64); // 最多 64 线程 List<Future<?>>…...
golang-ErrGroup用法以及源码解读笔记
介绍 ErrGroup可以并发执行多个goroutine,并可以很方便的处理错误 与sync.WaitGroup相比 错误处理 sync.WaitGroup只负责等待goroutine执行完成,而不处理返回值或者错误errgroup.Group目前虽然不能直接处理函数的返回值或错误。但是当goroutine返回错…...
遗传算法组卷系统实现(Java版)
遗传算法组卷系统实现(Java版) 下面是一个完整的遗传算法组卷系统的Java实现,包含题目表示、适应度计算、选择、交叉和变异等核心操作。 1. 核心类设计 1.1 题目实体类(Question.java) public class Question {pri…...
JVM 逃逸分析
JVM逃逸分析:让Java对象"乖乖听话"的侦探游戏 大家好!我是你们的老朋友"代码神探福尔摩斯",今天要带大家破解一个JVM界的经典悬案——如何让Java对象乖乖待在它该待的地方? 第一章:案发现场——…...
第三天——贪心算法——区间问题
题目描述 给定多个区间,计算让这些区间互不重叠所需要移除区间的最少个数。起止相连不算重叠。 输入输出样例 输入是一个数组,包含多个长度固定为的子数组,表示每个区间的开始和结尾。输出一个 整数,表示需要移除的区间数量。 …...
路由重发布
路由重发布 实验目标: 掌握路由重发布的配置方法和技巧; 掌握通过路由重发布方式实现网络的连通性; 熟悉route-pt路由器的使用方法; 实验背景:假设学校的某个分区需要配置简单的rip协议路由信息,而主校…...
嵌入式与物联网:C 语言在边缘计算时代的破局之道
引言 在万物互联的 2025 年,全球物联网设备连接数突破 300 亿台,其中 78% 的嵌入式控制系统仍基于 C 语言开发。这种跨越半个世纪的编程语言,正以新的技术形态在智能汽车、工业物联网、边缘计算等领域重塑竞争力。本文通过三个前沿应用场景&…...
计算机网络:WiFi路由器发射的电磁波在空气中的状态是什么样的?
WiFi路由器发射的电磁波是高频无线电波,属于微波频段(2.4GHz或5GHz),在空气中以光速传播(约310⁸米/秒),其传播状态和特性可通过以下维度详细解析: 一、电磁波的物理特性 频率与波长 2.4GHz频段:波长约12.5厘米,穿透力较强但易受干扰(微波炉、蓝牙等共用频段)。5GH…...
使用FastAPI和Apache Flink构建跨环境数据管道
系统概述 本文介绍如何使用FastAPI微服务、Apache Flink和AWS ElastiCache Redis构建一个可扩展的数据管道,实现本地Apache Hive数据仓库与AWS云上Redis之间的数据交互。 该架构通过FastAPI提供RESTful接口,Apache Flink处理数据流,实现了本…...
解决 SQL Server 2008 导入 Excel 表卡在“正在初始化数据流”问题
在使用 SQL Server 2008 导入 Excel 表时,可能会遇到卡在“正在初始化数据流”这一令人困扰的情况。笔者近期也遭遇了同样的问题,尝试了多种常规方法均未解决,最终通过特定命令成功化解难题,在此分享解决过程与经验。 一、问题描…...
【Linux系统】从零开始构建简易 Shell:从输入处理到命令执行的深度剖析
文章目录 前言一、打印命令行提示符代码功能概述 二、读取键盘输入的指令2.1 为什么不继续使用scanf()而换成了fgets()?2.2 调试输出的意义2.3 为什么需要去掉换行符? 三、指令切割补充知识: strtok 的函数原型 四、普通命令的执行代码功能概…...
SSRF服务端请求伪造
SSRF:服务端请求伪造 危害:任意文件读取、任意服务探测(通过端口来探测) 例:探测3306端口,看mysql服务是否开启,再通过文件读取,获得mysql配置文件 例:当我们点击链接…...
LVGL的三层屏幕结构
文章目录 🌟 LVGL 的三层屏幕架构1. **Top Layer(顶层)**2. **System Layer(系统层)**3. **Active Screen(当前屏幕层)** 🧠 总结对比🔍 整体作用✅ 普通屏幕层对象&…...
使用互斥锁保护临界
Linux线程互斥及相关概念解析 1. 临界资源(Critical Resource) 定义:被多个线程共享的资源(如变量、文件、内存区域等),需通过互斥访问确保数据一致性。特点: 共享性:多个线程可能…...
5.8线性动态规划2
P1004 [NOIP 2000 提高组] 方格取数 做法1:DFS剪枝 #include<bits/stdc.h> using namespace std; int n, a[10][10], maxs, minx 11, miny 11, maxx, maxy; void dfs(int x, int y, int s, int type){if(type 1 && x minx && y miny){…...
linux系统Ubuntn界面更改为中文显示,配置流程
Linux 系统是一种开源的、多用户的、多任务的操作系统,具有高度的稳定性、安全性和灵活性,被广泛应用于服务器、嵌入式系统、科研、教育以及个人电脑等领域。以下是关于 Linux 系统的一些基本信息: 发展历程:Linux 的发展始于 19…...
Looper死循环阻塞为什么没有ANR
Looper 死循环阻塞没有 ANR 的原因在于 ANR (Application Not Responding) 的检测机制依赖于特定线程的事件处理超时。以下是详细解释: 1. ANR 的触发机制: 主线程 (UI 线程) 阻塞: ANR 最常见的情况是主线程阻塞。Android 系统会监控主线程…...
数字孪生陆上风电场可视化管理系统
图扑软件搭建陆上风电场数字孪生平台,通过高精度建模与实时数据采集,1:1 还原风机设备、输电网络及场区环境。动态展示风机运行参数、发电量、设备健康状态等信息,实现风电场运维管理的智能化、可视化与高效化。...
图像处理篇---MJPEG视频流处理
文章目录 前言一、MJPEG流基础概念MJPEG流特点格式简单无压缩时序HTTP协议传输边界标记 常见应用场景IP摄像头视频流嵌入式设备(如ESP32)视频输出简单视频监控系统 二、基础处理方法方法1:使用OpenCV直接读取优点缺点 方法2:手动解…...
ensp的华为小实验
1.先进行子网划分 2.进行接口的IP地址配置和ospf的简易配置,先做到全网小通 3.进行ospf优化 对区域所有区域域间路由器进行一个汇总 对区域1进行优化 对区域2.3进行nssa设置 4.对ISP的路由进行协议配置 最后ping通5.5.5.5...
webpack和vite区别
webpack将文件视为模块打包 ,从入口文件递归解析依赖,生成依赖图,使用loader处理非JS模块,最终输出到dist目录 因为要解析所有依赖,所以他启动慢 vite利用浏览器对于es模块的原生支持,利用ESM能力&#x…...
从父类到子类:C++ 继承的奇妙旅程(2)
前言: 各位代码航海家,欢迎回到C继承宇宙!上回我们解锁了继承的「基础装备包」,成功驯服了public、protected和花式成员隐藏术。但—— ⚠️前方高能预警: 继承世界的暗流涌动远不止于此!今天我们将勇闯三大…...
ScaleTransition 是 Flutter 中的一个动画组件,用于实现缩放动画效果。
ScaleTransition 是 Flutter 中的一个动画组件,用于实现缩放动画效果。它允许你对子组件进行动态的缩放变换,从而实现平滑的动画效果。ScaleTransition 通常与 AnimationController 和 Tween 一起使用,以控制动画的开始、结束和过渡效果。 基…...
部署RocketMQ
部署环境:jdk8以上,Linux系统 下载和安装指令: wget https://archive.apache.org/dist/rocketmq/4.9.4/rocketmq-all-4.9.4-bin-release.zip 显示下载成功: --2025-05-10 11:34:46-- https://archive.apache.org/dist/rocketm…...
从爬虫到网络---<基石9> 在VPS上没搞好Docker项目,把他卸载干净
1.停止并删除所有正在运行的容器 docker ps -a # 查看所有容器 docker stop $(docker ps -aq) # 停止所有容器 docker rm $(docker ps -aq) # 删除所有容器如果提示没有找到容器,可以忽略这些提示。 2.删除所有镜像 docker images # 查看所有镜像 dock…...
每日c/c++题 备战蓝桥杯(P2241 统计方形(数据加强版))
洛谷P2241 统计方形(数据加强版)题解 题目描述 给定一个 n m n \times m nm 的方格棋盘,要求统计其中包含的正方形数量和长方形数量(不包含正方形)。输入为两个正整数 n n n 和 m m m,输出两个整数分…...
LLaVA:开源多模态大语言模型深度解析
一、基本介绍 1.1 项目背景与定位 LLaVA(Large Language and Vision Assistant)是由Haotian Liu等人开发的开源多模态大语言模型,旨在实现GPT-4级别的视觉-语言交互能力。该项目通过视觉指令微调技术,将预训练的视觉编码器与语言模型深度融合,在多个多模态基准测试中达到…...
基于Spring Boot + Vue的母婴商城系统( 前后端分离)
一、项目背景介绍 随着母婴行业在互联网平台的快速发展,越来越多的家庭倾向于在线选购母婴产品。为了提高商品管理效率和用户购物体验,本项目开发了一个基于 Spring Boot Vue 技术栈的母婴商城系统,实现了商品分类、商品浏览、资讯展示、评…...
HNUST湖南科技大学-软件测试期中复习考点(保命版)
使用说明:本复习考点仅用于及格保命。软件测试和其他专业课不太一样,记忆的太多了,只能说考试的时候,想到啥就写啥,多写一点!多写一点!多写一点!(重要事情说三遍…...
【AI智能推荐系统】第七篇:跨领域推荐系统的技术突破与应用场景
第七篇:跨领域推荐系统的技术突破与应用场景 提示语:🔥 “打破数据孤岛,实现1+1>2的推荐效果!深度解析美团、亚马逊如何用跨领域推荐技术实现业务协同,知识迁移核心技术全公开!” 目录 跨领域推荐的商业价值跨领域推荐技术体系 2.1 基于共享表征的学习2.2 迁移学习…...
【现代深度学习技术】注意力机制04:Bahdanau注意力
【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈PyTorch深度学习 ⌋ ⌋ ⌋ 深度学习 (DL, Deep Learning) 特指基于深层神经网络模型和方法的机器学习。它是在统计机器学习、人工神经网络等算法模型基础上,结合当代大数据和大算力的发展而发展出来的。深度学习最重…...
使用FastAPI和React以及MongoDB构建全栈Web应用01 概述
Are you ready to craft digital experiences that captivate and convert? 您准备好打造令人着迷并能带来转变的数字体验了吗? In a world driven by innovation, the demand for robust and scalable web applications has never been higher. Whether you’re…...
Flutter - UIKit开发相关指南 - 概览
环境 Flutter 3.29 macOS Sequoia 15.4.1 Xcode 16.3 概览 UIView与Widgets的比较 在UIKit使用UIView类的对象进行页面开发,布局也是UIView类的对象,在Flutter中使用的是Widget,在概念上Widget可以理解成UIView。 差异: 有效期: Widgets是不可变的,它的生存期只…...
扩容 QCOW2 磁盘镜像文件
🌈 个人主页:Zfox_ 目录 ✅ 一、扩展 QCOW2 文件大小✅ 二、启动虚拟机后扩展分区和文件系统方式一:如果使用的是标准分区(如 /dev/vda1)方式二:使用 gparted(图形工具) ✅ 总结 &am…...
【ts】for in对象时,ts如何正确获取对应的属性值
第一种:for…in keyof:适合需要遍历对象属性键并动态访问值的场景。 keyof typeof obj是ts的类型操作符,用于获取对象obj的所有属性键的联合类型(“name” | “age” | “city”)通过obj[key keyof typeof obj]&…...
软考 系统架构设计师系列知识点之杂项集萃(55)
接前一篇文章:软考 系统架构设计师系列知识点之杂项集萃(54) 第89题 某软件公司欲开发一个Windows平台上的公告板系统。在明确用户需求后,该公司的架构师决定采用Command模式实现该系统的界面显示部分,并设计UML类图如…...
绑定 SSH key(macos)
在 macOS 上绑定 Gitee 或 GitHub 的 SSH Key,通常分为以下几步操作,包括生成 SSH key、添加到 ssh-agent,并配置到 Gitee 或 GitHub 平台。 1. 检查是否已有 SSH Key ls -al ~/.ssh 看看是否已有 id_rsa 或 id_ed25519 等文件。如果没有就…...
PyTorch API 6 - 编译、fft、fx、函数转换、调试、符号追踪
文章目录 torch.compiler延伸阅读 torch.fft快速傅里叶变换辅助函数 torch.func什么是可组合的函数变换?为什么需要可组合的函数变换?延伸阅读 torch.futurestorch.fx概述编写转换函数图结构快速入门图操作直接操作计算图使用 replace_pattern() 进行子图…...
Unreal 从入门到精通之VR常用操作
文章目录 前言1.如何设置VRPawn视角的位置。2.如何播放视频3.如何播放VR全景视频。4.如何打开和关闭VR模式。前言 我们使用Unreal5 开发VR 项目的时候,会遇到很多常见问题。 比如: 1.如何设置VRPawn视角的位置。 2.如何播放视频。 3.如何播放VR全景视频。 4.如何打开和关闭V…...
Dify使用总结
最近完成了一个Dify的项目简单进行总结下搭建服务按照官方文档操作就行就不写了。 进入首页之后由以下组成: 探索、工作室、知识库、工具 探索: 可以展示自己创建的所有应用,一个应用就是一个APP,可以进行测试使用 工作室包含…...
事务连接池
一、事务概述 (一)事务的定义 事务是数据库提供的一种特性,用于确保数据操作的完整性和一致性。事务将多个数据操作组合成一个逻辑单元,这些操作要么全部成功,要么全部失败。 (二)事务的特性…...
如何用AWS Lambda构建无服务器解决方案:实战经验与场景解析
一、为什么开发者都在关注Serverless? 一、为什么开发者都在关注Serverless? 在云计算高速发展的今天,“无服务器架构”正成为技术新宠。根据Gartner预测,到2025年全球将有50%企业采用Serverless技术。而作为无服务器领域的领头…...
Android Compose 框架物理动画之捕捉动画深入剖析(29)
Android Compose 框架物理动画之捕捉动画深入剖析 一、引言 在 Android 应用开发中,动画是提升用户体验的关键元素之一。它能够让界面更加生动、交互更加自然。Android Compose 作为新一代的声明式 UI 框架,为开发者提供了强大且灵活的动画能力。其中&…...
Jmeter中的Json提取器如何使用?
在JMeter中使用JSON提取器可以方便地从JSON格式的响应数据中提取特定字段的值。以下是详细步骤和示例: 1. 添加JSON提取器 右击目标HTTP请求 -> 选择 添加 -> 后置处理器 -> JSON提取器。 2. 配置JSON提取器参数 变量名称(Names of created…...
STM32中断
STM32 GPIO外部中断简图 中断向量表 定义一块固定的内存,以4字节对齐,存放各个中断服务函数程序的首地址 中断向量表定义在启动文件,当发生中断,CPU会自动执行对应的中断服务函数 中断向量表以及中断函数 NVIC嵌套向量中断控制…...
navicat 如何导出数据库表 的这些信息 字段名 类型 描述
navicat 如何导出数据库表 的这些信息 字段名 类型 描述 数据库名字 springbootmt74k 表名字 address SELECT COLUMN_NAME AS 字段名,COLUMN_TYPE AS 类型,COLUMN_COMMENT AS 描述 FROM information_schema.COLUMNS WHERE TABLE_SCHEMA springbootmt74k AND TABLE_NAME a…...
LangGraph(三)——添加记忆
目录 1. 创建MemorySaver检查指针2. 构建并编译Graph3. 与聊天机器人互动4. 问一个后续问题5. 检查State参考 1. 创建MemorySaver检查指针 创建MemorySaver检查指针: from langgraph.checkpoint.memory import MemorySavermemory MemorySaver()这是位于内存中的检…...
数仓-可累计,半累加,不可累加指标,是什么,举例说明及解决方案
目录 1. 可累计指标定义:举例:解决方案: 2. 半累加指标定义:举例:解决方案: 3. 不可累加指标定义:举例:解决方案: 4. 总结对比5. 实际场景中的注意事项 这是数据仓库设计…...
Java ClassLoader双亲委派机制
Java ClassLoader双亲委派机制 1 什么是双亲委派模型 “类加载体系”及ClassLoader双亲委派机制。java程序中的 .java文件编译完会生成 .class文件,而 .class文件就是通过被称为类加载器的ClassLoader加载的,而ClassLoder在加载过程中会使用“双亲委派…...
upload-labs靶场通关详解:第四关
一、分析源代码 可以看出这一关仍然是黑名单验证,但是它禁止了更多的后缀。像php3,php4这类后缀也被加入了黑名单,第三关的方法在这里显然就失效了。那么我们想一想,既然配置文件中存在将php3当作php来执行的功能,那么…...