ResponseEntity<Stream<MyObject>> 是否将所有内容都存储在内存中?

zib*_*ibi 4 java spring spring-mvc spring-boot

这似乎是一个简单的问题,但我找不到答案。

在 spring-boot 中,我有一个返回 json 的 Controller 方法:

@GetMapping
public ResponseEntity<Stream<MyObject>> get() {
  return ResponseEntity
    .ok()
    .contentType(MEDIA_JSON)
    .body(service.stream());
}

Run Code Online (Sandbox Code Playgroud)

这是正确地流式传输给用户(以块为单位)还是在返回给用户之前将整个流加载到内存中?

xon*_*nya 5

我在文档中没有找到任何内容,但是可以尝试查看它的行为方式。

让我们创建 2 个端点和一个无限的对象流:

@RestController
public class MyController {

    @GetMapping("/stream")
    public ResponseEntity<Stream<MyObject>> stream() {
        return ResponseEntity.ok(getStream());
    }

    @GetMapping("/nostream")
    public ResponseEntity<List<MyObject>> noStream() {
        return ResponseEntity.ok(getStream().collect(Collectors.toList()));
    }

    private Stream<MyObject> getStream() {
        return IntStream.iterate(0, i -> i + 1)
                .mapToObj(i -> new MyObject());
    }

    static class MyObject {

        private String name = "foo";

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

调用 http://localhost:8080/stream 将立即响应生成无限序列 [{"name":"foo"},{"name":"foo"}...

如果您分析应用程序,您可以看到它可以继续运行几分钟而不会增加内存使用量:

使用流进行分析

相反,调用 http://localhost:8080/nostream 端点将很快导致java.lang.OutOfMemoryError

没有流的分析