【Easylive】详细解析 `stream()` 方法的使用

【Easylive】项目常见问题解答(自用&持续更新中...) 汇总版

saveVideoInfo 方法中,stream() 方法被多次用于对集合进行函数式操作。下面我将详细解析每个 stream() 的使用场景和实现逻辑。

1. 将上传文件列表转为Map

java 复制代码
Map<String, VideoInfoFilePost> uploadFileMap = uploadFileList.stream()
    .collect(Collectors.toMap(
        item -> item.getUploadId(),  // Key映射函数
        Function.identity(),         // Value映射函数
        (data1, data2) -> data2      // 合并函数(解决键冲突)
    ));

解析:

目的 :将 List<VideoInfoFilePost> 转换为 Map<String, VideoInfoFilePost>,以便通过 uploadId 快速查找文件

分解

复制代码
1. `uploadFileList.stream()` - 将列表转为流
2. `Collectors.toMap()` - 收集器,将流元素转为Map
   ◦ 第一个参数 `item -> item.getUploadId()`:指定使用 `uploadId` 作为Map的键
   ◦ 第二个参数 `Function.identity()`:使用元素本身作为Map的值
   ◦ 第三个参数 `(data1, data2) -> data2`:如果键冲突(相同uploadId),保留后面的值

等价传统写法:

java 复制代码
Map<String, VideoInfoFilePost> uploadFileMap = new HashMap<>();
for (VideoInfoFilePost item : uploadFileList) {
    uploadFileMap.put(item.getUploadId(), item);
}

2. 筛选新增文件

java 复制代码
addFileList = uploadFileList.stream()
    .filter(item -> item.getFileId() == null)
    .collect(Collectors.toList());

解析:

目的 :从上传文件列表中筛选出新增的文件(fileId为null的文件)

分解

复制代码
1. `uploadFileList.stream()` - 创建流
2. `.filter(item -> item.getFileId() == null)` - 过滤条件
3. `.collect(Collectors.toList())` - 收集为List

等价传统写法:

java 复制代码
List<VideoInfoFilePost> addFileList = new ArrayList<>();
for (VideoInfoFilePost item : uploadFileList) {
    if (item.getFileId() == null) {
        addFileList.add(item);
    }
}

3. 提取待删除文件的ID列表

java 复制代码
List<String> delFileList = deleteFileList.stream()
    .map(item -> item.getFileId())
    .collect(Collectors.toList());

解析:

目的 :从待删除文件列表中提取所有 fileId 组成新列表

分解

复制代码
1. `deleteFileList.stream()` - 创建流
2. `.map(item -> item.getFileId())` - 映射转换,从对象中提取 `fileId`
3. `.collect(Collectors.toList())` - 收集为List

等价传统写法:

java 复制代码
List<String> delFileList = new ArrayList<>();
for (VideoInfoFilePost item : deleteFileList) {
    delFileList.add(item.getFileId());
}

4. 提取待删除文件的路径列表

java 复制代码
List<String> delFilePathList = deleteFileList.stream()
    .map(item -> item.getFilePath())
    .collect(Collectors.toList());

解析:

目的 :从待删除文件列表中提取所有 filePath 组成新列表

实现 :与上一个例子类似,只是提取的是 filePath 而非 fileId

Stream API 核心概念总结

操作类型 方法 说明 示例
创建流 stream() 从集合创建流 list.stream()
中间操作 filter() 过滤元素 .filter(x -> x > 0)
map() 转换元素 .map(x -> x.getName())
终止操作 collect() 收集结果 .collect(Collectors.toList())
forEach() 遍历元素 .forEach(System.out::println)

为什么使用Stream API?

  1. 代码简洁:用声明式的方式表达复杂的数据处理逻辑
  2. 可读性强:链式调用更直观表达数据处理流程
  3. 并行处理 :只需将stream()改为parallelStream()即可实现并行
  4. 惰性求值:中间操作不会立即执行,提高效率

性能考虑

虽然Stream API很强大,但在简单循环就能完成的任务中,传统for循环可能更高效。Stream适合处理复杂的数据转换和过滤场景。

完整流程示例

假设有以下数据:

java 复制代码
List<VideoInfoFilePost> uploadFileList = Arrays.asList(
    new VideoInfoFilePost("f1", "u1", "p1"), // fileId不为null,视为已有文件
    new VideoInfoFilePost(null, "u2", "p2"), // fileId为null,视为新增文件
    new VideoInfoFilePost(null, "u3", "p3")  // fileId为null,视为新增文件
);

执行:

java 复制代码
List<VideoInfoFilePost> addFileList = uploadFileList.stream()
    .filter(item -> item.getFileId() == null)
    .collect(Collectors.toList());

结果:

java 复制代码
addFileList = [
    VideoInfoFilePost(null, "u2", "p2"),
    VideoInfoFilePost(null, "u3", "p3")
]

通过Stream API,我们简洁地实现了数据筛选和转换,代码更加清晰易读。

相关推荐
佳xuan几秒前
wsl(linux)安装miniconda及虚拟环境
linux·人工智能·conda
召田最帅boy3 分钟前
一次OOM排查实录
linux·jvm·spring boot·adb
无巧不成书02187 分钟前
Windows环境变量故障排查:记事本BOM头导致配置失效终极解决方案 | 零基础全流程指南
windows·批处理脚本故障·windows故障排查·windows记事本·bom头·utf-8 bom·环境变量读取失效 环境变量配置
The_cute_cat9 分钟前
CentOS Stream 10虚拟机固定ip总结
linux·运维·centos
一只积极向上的小咸鱼19 分钟前
Vscode打开多个窗口
linux·运维·服务器·vscode
IMPYLH21 分钟前
Linux 的 shred 命令
linux·运维·服务器·bash
咖喱o24 分钟前
ARP代理(ARP Proxy)
运维·服务器·网络
charlie11451419128 分钟前
嵌入式Linux开发(6-前置)——IDE 配置指南 - VSCode + clangd 驱动开发环境搭建
linux·ide·驱动开发·vscode
FuckPatience29 分钟前
Visual Studio的配置管理器
windows·visual studio