【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,我们简洁地实现了数据筛选和转换,代码更加清晰易读。

相关推荐
唯独失去了从容1 小时前
WebRTC服务器Coturn服务器的管理平台功能
运维·服务器·webrtc
PassLink_5 小时前
[Kaggle]:使用Kaggle服务器训练YOLOv5模型 (白嫖服务器)
运维·服务器·yolo
朴拙数科5 小时前
MongoDB Atlas与MongoDB连接MCP服务器的区别解析
服务器·数据库·mongodb
爱编程的鱼6 小时前
C# 枚举(Enum)声明与使用详解
java·windows·c#
程序猿(雷霆之王)6 小时前
Linux——进程间通信
linux·运维·服务器
riveting7 小时前
SD2351核心板:重构AI视觉产业价值链的“超级节点”
大数据·linux·图像处理·人工智能·重构·智能硬件
易保山8 小时前
MIT6.S081 - Lab10 mmap(文件&内存映射)
linux·操作系统·c
NoneCoder8 小时前
HTML 模板技术与服务端渲染
服务器·servlet·html
禅与Bug的修复艺术8 小时前
JAVA后端开发常用的LINUX命令总结
java·linux·面试·java面试·后端开发·java后端·面试经验
李詹8 小时前
Steam游戏服务器攻防全景解读——如何构建游戏级抗DDoS防御体系?
服务器·游戏·ddos