为什么我们从代码中放弃反应式系统架构?
技术百科
WBOY
发布时间:2024-08-29
浏览: 次 本文探讨了我们在软件项目中放弃反应式架构的决定。我们将深入研究反应式系统的核心原则、非阻塞 i/o 的好处以及反应式方法所面临的挑战。
理解响应式架构风格
reactive 包含一系列旨在构建响应式分布式系统和应用程序的原则和指南,其特点是:
- 响应能力:即使在重负载下也能够快速处理请求。
- 弹性:能够以最短的停机时间从故障中恢复。
- 弹性:可以通过相应地扩展资源来适应不断变化的工作负载。
- 消息驱动:利用异步消息传递来增强容错能力并解耦组件。
反应式系统的一个主要好处是它们使用非阻塞 i/o。这种方法避免了 i/o 操作期间阻塞线程,允许单个线程同时处理多个请求。与传统的阻塞 i/o 相比,这可以显着提高系统效率。
在传统的多线程中,阻塞操作对优化系统提出了重大挑战(图 1)。消耗过多内存的贪婪应用程序效率低下,并且会惩罚其他应用程序,通常需要请求额外的资源,如内存、cpu 或更大的虚拟机。
图 1 – 传统多线程
i/o 操作是现代系统不可或缺的一部分,有效管理它们对于防止贪婪行为至关重要。反应式系统采用非阻塞 i/o,使少量的操作系统线程能够处理大量并发 i/o 操作。
反应式执行模型
虽然非阻塞 i/o 提供了巨大的好处,但它引入了一种不同于传统框架的新颖的执行模型。响应式编程的出现就是为了解决这个问题,因为它可以缓解阻塞操作期间平台线程空闲的低效率问题(图 2)。
图 2 – 响应式事件循环
quarkus 和 reactive
quarkus 利用由 eclipse vert.x 和 netty 提供支持的反应式引擎,促进非阻塞 i/o 交互。 mutiny 是使用 quarkus 编写反应式代码的首选方法,它采用事件驱动范例,其中反应由接收到的事件触发。
mutiny 提供两种事件驱动和惰性类型:
- uni: 发出单个事件(一项或失败),适合表示具有零个或一个结果的异步操作。
- multi: 发出多个事件(n 个项目、一个失败或一个完成),表示项目流,可能是无限的。
响应式的挑战
虽然反应式系统提供了好处,但我们在开发过程中遇到了一些挑战:
- 范式转变: 响应式编程需要开发人员思维方式的根本转变,这可能具有挑战性,特别是对于习惯命令式编程的开发人员来说。与 streams api 等辅助工具不同,反应式方法需要彻底改变思维方式。
- 代码可读性和理解: 反应式代码给新开发人员带来了理解困难,导致破译和理解代码的时间增加。反应式范式带来的复杂性加剧了这个问题。
“事实上,阅读与写作所花费的时间之比远远超过 10 比 1。我们不断地阅读旧代码,作为编写新代码的一部分。...[因此,]使其易于阅读使得写起来更容易。” ― robert c. martin,《整洁代码:敏捷软件工艺手册》
- 调试挑战:由于 lambda 封装了大多数代码,使用标准 ide 调试器调试反应式代码几乎是不可能的。此外,异常期间有意义的堆栈跟踪的丢失进一步阻碍了调试工作。 增加开发和测试工作:由于编写、修改和测试所需的时间,反应式代码固有的复杂性可能会导致更长的开发周期。
这是一个使用 mutiny 的反应式代码示例来说明其复杂性:
Multi.createFrom().ticks().every(Duration.ofSeconds(15))
.onItem().invoke(() - > Multi.createFrom().iterable(configs())
.onItem().transform(configuration - > {
try {
return Tuple2.of(openAPIConfiguration,
RestClientBuilder.newBuilder()
.baseUrl(new URL(configuration.url()))
.build(MyReactiveRestClient.class)
.getAPIResponse());
} catch (MalformedURLException e) {
log.error("Unable to create url");
}
return null;
}).collect().asList().toMulti().onItem().transformToMultiAndConcatenate(tuples - > {
AtomicInteger callbackCount = new AtomicInteger();
return Multi.createFrom().emitter(emitter - > Multi.createFrom().iterable(tuples)
.subscribe().with(tuple - >
tuple.getItem2().subscribe().with(response - > {
emitter.emit(callbackCount.incrementAndGet());
if (callbackCount.get() == tuples.size()) {
emitter.complete();
}
})
));
}).subscribe().with(s - > {},
Throwable::printStackTrace, () - > doSomethingUponComplete()))
.subscribe().with(aLong - > log.info("Tic Tac with iteration: " + aLong));
未来展望-project loom 及未来
project loom 是 java 生态系统的最新开发项目,有望缓解与阻塞操作相关的问题。通过在不更改硬件的情况下创建数千个虚拟线程,project loom 可能在许多情况下消除对反应式方法的需求。
“loom 项目将杀死响应式编程”
―布莱恩·戈茨
结论
总之,我们决定放弃反应式架构风格,采用务实的方法来实现项目的长期可维护性。虽然反应式系统提供了潜在的好处,但它们给我们团队带来的挑战超过了我们特定环境中的这些优势。
重要的是,这种转变并没有影响性能。这是一个积极的结果,因为它表明设计良好的非反应式(命令式)架构可以提供必要的性能,而不会带来与我们案例中的反应式架构相关的复杂性。
展望未来,我
们的重点仍然是构建一个代码库,该代码库不仅实用,而且易于所有经验水平的开发人员理解和维护。这不仅减少了开发时间,还促进了团队内更好的协作和知识共享。
在下图中,x 轴 代表我们的代码库在发展过程中不断增加的复杂性,而 y 轴 则描述了这些开发变化所需的时间。
# 应用程序
# 过程中
# 思维方式
# 多个
# 所需
# 这可
# 这是一个
# 系统架构
# 因为它
# 循环
# 并发
# 堆
# Java
# 线程
# 架构
# 栈
# 异步
# 事件
# 多线程
# 封装
# 开发人员
# Lambda
# ide
# 分布式
# 代码可读性
# eclipse
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11如何设置鼠标灵敏度_Win11鼠标灵敏度
- Win11怎样安装剪映专业版_Win11安装剪映教
- Golang如何避免指针逃逸_Golang逃逸分析
- Win11如何更新显卡驱动 Win11检查和安装设
- Win11鼠标灵敏度怎么调 Win11鼠标指针移动
- Drupal 中 HTML 链接被重复转义导致渲染
- Windows 10自带杀毒软件在哪_Window
- C++如何编写函数模板?(泛型编程入门)
- Windows音频驱动无声音原因解析_声卡驱动错误
- 如何使用Golang benchmark测量函数延
- 如何高效识别并拦截拼接式恶意域名 spam
- MySQL 中使用 IF 和 CASE 实现查询字
- 如何使用Golang配置安全开发环境_防止敏感信息
- Win11更新后变慢怎么办_Win11系统更新后卡
- C++如何获取CPU核心数?(std::threa
- 如何在 Go 中可靠地测试含 time.Time
- php怎么下载安装后设置默认字符集_utf8配置步
- C#如何使用Channel C#通道实现异步通信
- C#怎么使用委托和事件 C# delegate与e
- Win11怎么设置声音输出设备_Windows11
- c++怎么实现大文件的分块读写_c++ 文件指针s
- Win10怎样安装PPT模板_Win10安装PPT
- 如何使用Golang反射创建map对象_动态生成键
- 如何有效拦截拼接式恶意域名的垃圾信息
- Python 中将 ISO 8601 时间戳转换为
- Win11怎么关闭自动维护 Win11禁用系统自动
- 如何使用Golang搭建本地API测试环境_快速验
- c++ std::atomic如何保证原子性 c+
- 如何使用Golang template生成文本模板
- Go 中 := 短变量声明的类型推导机制详解
- Python安全爬虫设计_IP代理池与验证码识别策
- Mac怎么设置鼠标滚动速度_Mac鼠标设置详细参数
- 如何在Golang中处理二进制数据_Golang
- 如何使用Golang编写单元测试_创建Test函数
- Win11怎么设置默认输入法 Win11固定中文输
- 如何使用Golang开发基础文件下载功能_Gola
- 如何快速验证Golang安装是否成功_运行go v
- 如何在Golang中实现RPC异步返回_Golan
- VSC怎么快速定位PHP错误行_错误追踪设置法【方
- c++20的std::format怎么用 比pri
- c++中如何对数组进行排序_c++数组排序算法汇总
- 如何使用Golang反射将map转换为struct
- c++ std::future和std::prom
- Win11怎么更改计算机名_Windows11系统
- Win10 BitLocker加密教程 Win10
- c++输入输出流 c++ cin与cout格式化输
- 如何在Golang中实现并发消息队列消费者_Gol
- c++如何连接Redis c++ hiredis库
- C#如何使用XPathNavigator高效查询X
- Win11怎么更改系统语言_Win11中文语言包下

QQ客服