// 获取所有红色苹果的总重量
appleStore.stream().filter(a -> "red".equals(a.getColor()))
.mapToInt(w -> w.getWeight()).sum()
// 基于颜色统计平均重量
appleStore.stream().collect(Collectors.groupingBy(a -> a.getColor(),
Collectors.averagingInt(a -> a.getWeight()))).forEach((k, v) -> {
System.out.println(k + ":" + v);
});
场景一:跨库join的问题
查询一个店铺的订单信息,需要用到订单表与会员表 在传统数据库单一例中 可以通过jon 关联轻松实现,但在分布场景中 这两张表分别存储在于 交易库 和会员库 两个实例中,join不能用。只能在服务端实现其流程如下:
这用传统迭代方法非常繁琐,而这正是stream 所擅长的。示例代码如下:
// 获取所有会员ID 并去重
List<Integer> ids = orders.stream().map(o -> o.getMemberId()).distinct().collect(Collectors.toList());
// 合并会员信息 至订单信息
orders.stream().forEach(o -> {
Member member = members.stream().filter(m -> m.getId() == o.getMemberId()).findAny().get();
o.setMemberName(member.getName());
});
场景二:N+1 问题
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("hello world");
}
}).start();
Lambda写法
new Thread(() -> System.out.println("hello world")).start();
在上述例子中编译器会将 “System.out.println(“hello world”)” 编译成Runnable.run 的执行指令。因为 run 方法是Runnable接口的唯一方法,代码中我们无需指明Run方法。如果Runable有多个方法是不能使用Lambda表达示的。
支持Lambda的接口统称函数式接口。
只有一个抽像方法的接口就是函数式接口,其详细特征如下:
stream 所有操作组合在一起即变成了管道,管道中有以下两种操作:
方法 | 描述 | 操作类型 |
---|---|---|
filter | 接收一个Boolean表达示来过滤元素 | 中间操作 |
map | 将流中元素 1:1 映谢成另外一个元素 | 中间操作 |
mapToInt | 将流中元素映谢成int,mapToLong、mapToDouble操作类似目的减少 装箱拆箱带来的损耗 | 中间操作 |
flatMap | 如map时返回的是一个List, 将会进一步拆分。详见flatMap示例 | 中间操作 |
forEach | 遍历流中所有元素 | 终值操作 |
sorted | 排序 | 中间操作 |
peek | 遍历流中所有元素 ,如forEach不同在于不会结束流 | 中间操作 |
toArray | 将流中元素转换成一个数组返回 | 终值操作 |
reduce | 归约合并操作 | 中间操作 |
collect | 采集数据,返回一个新的结果 参数说明:Supplier: 采集需要返回的结果BiConsumer<R, ? super T>:传递结果与元素进行合并。BiConsumer<R, R>:在并发执行的时候 结果合并操作。详见 collec示例 | 终值操作 |
distinct | 基于equal 表达示去重 | 中间操作 |
max | 通过比较函数 返回最大值 | 终值操作 |
anyMatch | 流中是否有任一元素满足表达示 | 终值操作 |
allMatch | 流中所有元素满足表达示返回true | 终值操作 |
noneMatch | 与allMatch 相反,都不满足的情况下返回 true | 终值操作 |
findFirst | 找出流中第一个元素 | 终值操作 |
of | 生成流 | 生成流操作 |
iterate | 基于迭代生成流 | 生成流操作 |
generate | 基于迭代生成流,与iterate 不同的是不 后一元素的生成,不依懒前一元素 | 生成流操作 |
concat | 合并两个相同类型的类 | 生成流操作 |
举例:
@Test
public void filterTest() {
appleStore.stream().filter(a -> a.getColor().equals("red")).forEach(a -> {
System.out.println(a.getColor());
});
}
@Test
public void mapTest() {
appleStore.stream().map(a -> a.getOrigin()).forEach(System.out::println);
}
@Test
public void flatMapTest() throws IOException {
Stream<String> lines = Files.lines(new File("G:\\git\\tuling- java8\\src\\main\\java\\com\\tuling\\java8\\stream\\bean\\Order.java").toPath());
lines.flatMap(a -> Arrays.stream(a.split(" "))).forEach(System.out::println);
}
@Test
public void sortedTest() {
appleStore.stream().sorted((a, b) -> a.getWeight() - b.getWeight())
.map(a -> a.getWeight()).forEach(System.out::println);
}
@Test
public void peekTest() {
appleStore.stream().peek(a -> {
System.out.println(a.getId());
}).map(a -> a.getOrigin())
.peek(System.out::println).forEach(a -> {
});
}
@Test
public void reduceTest() {
// 找出最重的那个苹果
appleStore.stream().reduce((a, b) -> a.getWeight() > b.getWeight() ? a : b)
.ifPresent(a -> {
System.out.println(a.getWeight());
});
}
@Test
public void collectTest() {
// 将结果转换成id作为key map<Integer,Apple>
HashMap<Integer, Apple> map = appleStore.stream().collect(HashMap::new, (m, a) -> m.put(a.getId(), a), (m1, m2) -> m1.putAll(m2));
map.forEach((k, v) -> {
System.out.println(k);
System.out.println(v);
});
// Map<String,List<Apple>>
// 基于颜色分组, 并获取其平均重量
}
方法 | 描述 |
---|---|
toList | 转换成list |
toMap | 转换成map |
groupingBy | 统计分组 |
averagingInt | 求平均值 |
summingInt | 求总值 |
maxBy | 获取最大值 |
举例:
// 获得所有颜色苹果的平均重量
@Test
public void groupByTest() {
Collector<Apple, ?, Map<String, Double>> groupCollect =
Collectors.groupingBy((Apple a) -> a.getColor(), Collectors.averagingInt((Apple a) -> a.getWeight()));
appleStore.stream().collect(groupCollect).forEach((k, v) -> {
System.out.println(k + ":" + v);
});
}
一般情况使用完流之后不需要调用 close 方法进行关闭,除非是使用channel FileInputStream 这类的操作需要关闭,可调用 java.util.stream.BaseStream#onClose() 添加关闭监听。
版权说明 : 本文为转载文章, 版权归原作者所有 版权申明
原文链接 : https://blog.csdn.net/qq_43842093/article/details/121443028
内容来源于网络,如有侵权,请联系作者删除!