- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我们在使用Spring Boot 2.0、Webflux 5.0.7和Netty 4.1.25时出现网络传输问题。我们想要将 100000 个序列化为 JSON 的项目(大约 10Mb 的网络流量)传输到 1 个客户端。
NIO 和传统 IO 之间的网络传输性能有很大差异。测试结果如下:
Start reading 100000 from server in 5 iterations
Avg HTTP 283 ms
Avg stream 8130 ms
目前每秒请求数不是问题,但网络传输速度才是问题。我们已经读到,就网络速度而言,NIO 可能会慢大约 30%,但 1/30x 是大材小用。
在客户端和服务器端进行采样时,我们发现原因主要在于服务器端的实现。从下面的屏幕截图可以看出,服务器大部分时间都花在方法 select()
和 doWrite()
上。
端点代码本身:
@RestController
@RequestMapping(produces = {APPLICATION_JSON_VALUE, APPLICATION_STREAM_JSON_VALUE})
@Validated
public class StreamingController {
@GetMapping("/instruments/{eodDate}")
public Flux<TestItem> getInstruments(
@PathVariable @DateTimeFormat(pattern = "yyyy-MM-dd") LocalDate eodDate,
@RequestParam(required = false) Instant asOfTimestamp) {
//Generate test data in memory
List<TestItem> collect = IntStream.range(0, 100000)
.mapToObj(i -> new TestItem.Builder().build())
.collect(Collectors.toList());
return Flux.fromIterable(collect);
}
}
我们正在为 Netty 使用 Spring Boot 配置,我们怀疑默认情况下 Netty 配置错误。我们正在寻求您的帮助。我会根据您的要求添加任何其他详细信息。
更新:目标是分批读取整个响应以避免将所有响应放入内存,因为预期的数据量很大(几 Gb)。在客户端消费一批数据而不是一个元素是可以接受的。
最佳答案
您实际上并不是在测试 NIO 与 IO。 Spring WebFlux 应用程序始终在服务器级别使用非阻塞 IO(使用 Netty、Undertow 或任何 Servlet 3.1+ 异步 IO 兼容服务器)。
在这种情况下,您正在比较:
"application/json"
使用 Spring WebFlux 一次性完成负载"application/stream+json"
使用 Spring WebFlux 响应在第一种情况下,Spring WebFlux 以 react 方式生成响应主体,但将缓冲和刷新决策留给服务器本身。写入网络是有成本的,缓冲 but 和写入更大的 block 是有效的。
在第二种情况下,您要求 Spring WebFlux 为 Flux
的每个元素编写和刷新
。当客户端正在收听(可能是无限的)事件流并且两个不同事件之间可能有一些时间时,这很有用。这种方法会消耗更多资源并解释性能差异。
所以这个基准不是显示 IO 与 NIO,而是流式与非流式。
如果你想对响应写入/刷新进行细粒度控制,你可以下降到 ServerHttpResponse
级别并使用 writeAndFlushWith(Flux<Flux<DataBuffer>>)
,但这是相当低的级别,因为您正在处理 DataBuffer
直接实例。
另一种方法是创建包含 TestItem
列表的中间 JSON 对象,比如:
public Flux<TestItemBatch> batch() {
Flux<TestItem> items= //...;
Flux<List<TestItem>> itemsLists = items.buffer(100);
return itemsLists.map(list -> new TestItemBatch(list));
}
关于java - Webflux + Netty NIO 相比传统IO性能下降~30倍,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52837041/
我是 Redis 的新手,我现在正在尝试制作一个登录功能来学习它。 假设我在传统 SQL 数据库中有一个名为 User (id, username, password) 的表,设计 Redis 的正确
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题?通过 editing this post 添加详细信息并澄清问题. 5年前关闭。 Improve this
我刚刚遇到了具有所有天赐优势的 NoSQL 系统。其中之一似乎是毫不费力的水平缩放。我的问题是,为什么像 MySQL 或 SQL Server 这样的经典 RDBMS 不能进行水平扩展?或者无法做到与
问题 Although you can explicitly check if a value is true or false, it's a convention in JavaScript to
用于突发错误纠正的 ECC 如何工作? “突发错误检测”是指一种技术,可以检测(例如)任何一个 [或两个] 64 位连续位序列中的任何位错误组合。 我需要概念上的解释,而不是数学。 我研究了几种用无穷
我试图向学生说明传统 IO 和 java 中的内存映射文件之间的性能差异。我在互联网上的某个地方找到了一个例子,但我并不是很清楚所有的事情,我什至不认为所有的步骤都是必要的。我在这里和那里阅读了很多关
如何在 PhpStorm 中自动将短数组语法转换为长数组(传统)? 我使用了PhpStorm中的“代码->检查代码”功能,然后一键将短语法转换为长语法。 检查代码 手动转换 必须有一种方法来自动化这项
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我发现 Light Table 是尝试 Clojure 语言的好工具。 它有一个方便的功能叫做 Instarepl - ;;您在此处输入的任何内容都将立即执行,结果显示在右侧。 但是,我找不到使用旧式
我们有一个经典的 ASP 应用程序,所有数据都是使用表格呈现的(是的,我们正在努力将其转换为 DIV,但我们离那还有一段距离)。我们设置了一个标准的表格宽度,通常这可以正常工作,但有些元素的大小可以由
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 7 年前。 Improve
关闭。这个问题是opinion-based .它目前不接受答案。 想改进这个问题?更新问题,以便 editing this post 可以用事实和引用来回答它. 6年前关闭。 Improve this
这只是一个一般性问题。传统的 ajax 和 jquery ajax 哪个更好?因为当我使用 jquery ajax 时,它似乎在 IE 中做了一些奇怪的事情。它应该是一个跨浏览器库......所以我对
所有这些都是为了丢弃MVC Controller 的问题。 这是ajax的代码: $.ajax({ //tipo de transferencia t
我知道它可能应该是另一种方式(OSGI 运行时托管 Java EE 服务器),因为它是 apparently already possible with glassfish. 但是,我们的客户在管理传
作为具有一些 winforms 和客户端应用程序经验的人 - 是否值得回去学习传统的 ASP .NET 页面的工作方式,或者直接进入 ASP .NET MVC 是否可以? 我正在寻找我对 C# 的一般
如果我错了,请纠正我:在传统的Web应用程序中,浏览器会自动将 session 信息附加到对服务器的请求中,以便服务器可以知道该请求来自谁。实际上到底附加了什么? 但是,在基于 API 的应用程序中,
我正在使用Eclipse neon 3,当我导入一个使用WebSphere Application Server传统V9.0作为运行时环境的项目时,它显示了一个错误,宣布 Target runtime
背景: 我的 REST 服务项目是使用 Hibernate 启动的。我在域类中使用 id (Long) 作为 rest url 中标识符的一部分,例如: http://abc.com/customer
在 Scheme 中递归构建列表时,我看到两种类型的示例散布在互联网上。其中一个新值附加了 append每一次迭代。另一个在每次迭代前都添加一个新值 cons然后列表完成后reverse被调用一次。
我是一名优秀的程序员,十分优秀!