◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。
java streams 在 java 8 中引入,是该语言最强大的补充之一。它们支持对集合和序列进行函数式操作,改变了我们在 java 中处理数据的方式。流简化了过滤、映射和收集数据等任务,同时还支持并行操作以提高性能。在这篇文章中,我们将探讨 streams 的基础知识,讨论它们支持的操作类型,并提供示例来帮助您充分利用这一基本功能。
目录
1. what is streams and why we need it? 2. types of streams: intermediate vs. terminal 3. creating streams in java 4. intermediate stream operations 5. terminal stream operations 6. using streams with lambdas 7. conclusion
java 中的流提供了一种处理数据集合的强大方法。它们允许我们对集合的元素执行功能操作,例如过滤和转换,而无需改变底层数据。流帮助开发人员专注于他们想要实现的目标,而不是如何实现它,为数据处理提供了更高级别的抽象。
java 8 中引入了流以及 lambda 表达式和函数式接口,旨在使 java 更具表现力并减少样板代码。通过合并流,java 开始拥抱函数式编程范式,允许更干净、更简洁的代码。
立即学习“Java免费学习笔记(深入)”;
流的主要优点
流分为两种主要类型:
示例:
list<string> names = list.of("alice", "bob", "charlie", "david"); // intermediate (lazy) operations: filter and map stream<string> stream = names.stream() .filter(name -> name.startswith("a")) .map(string::touppercase); // terminal operation: collect list<string> filterednames = stream.collect(collectors.tolist()); system.out.println(filterednames); // output: [alice]
在这个例子中,filter和map是中间操作,直到调用终端操作collect后才会执行。
java 提供了多种创建流的方法,可以轻松开始处理数据。
创建流的最常见方法是使用 list、set 和 map 等集合。
list<string> names = list.of("alice", "bob", "charlie"); stream<string> namestream = names.stream();
string[] namesarray = {"alice", "bob", "charlie"}; stream<string> namestream = arrays.stream(namesarray);
stream<string> stream = stream.of("alice", "bob", "charlie");
java 允许使用 stream.generate 和 stream.iterate 创建无限流。
stream<double> randomnumbers = stream.generate(math::random).limit(5); stream<integer> counting = stream.iterate(0, n -> n + 1).limit(5);
中间操作返回一个新流并且是惰性的。这意味着它们仅在调用终端操作时执行。
根据条件过滤元素。
list<integer> numbers = list.of(1, 2, 3, 4, 5); list<integer> evennumbers = numbers.stream() .filter(n -> n % 2 == 0) .collect(collectors.tolist());
将元素从一种类型转换为另一种类型。
list<string> names = list.of("alice", "bob"); list<integer> namelengths = names.stream() .map(string::length) .collect(collectors.tolist());
按自然顺序或基于比较器对元素进行排序。
list<string> names = list.of("bob", "alice", "charlie"); list<string> sortednames = names.stream() .sorted() .collect(collectors.tolist());
对每个元素执行操作,通常对于调试很有用。
list<string> names = list.of("alice", "bob"); names.stream() .peek(name -> system.out.println("processing " + name)) .collect(collectors.tolist());
最后执行终端操作,触发实际的数据处理并返回最终结果。
对流中的每个元素执行一个操作。
list<string> names = list.of("alice", "bob"); names.stream().foreach(system.out::println);
将流的元素收集到集合、列表、集合或其他数据结构中。
list<string> names = list.of("alice", "bob"); set<string> nameset = names.stream().collect(collectors.toset());
计算流中元素的数量。
list<string> names = list.of("alice", "bob"); long count = names.stream().count();
检查是否有任何、全部或没有元素符合给定条件。
list<string> names = list.of("alice", "bob", "charlie"); boolean hasalice = names.stream().anymatch(name -> name.equals("alice"));
返回一个可选描述流的第一个或任何元素。
list<string> names = list.of("alice", "bob"); optional<string> first = names.stream().findfirst();
流和 lambda 表达式齐头并进。由于流基于函数式接口,因此它们可以与 lambda 无缝协作,从而实现富有表现力且简洁的数据处理。
例如,过滤名称列表以查找以“a”开头的名称,然后将其转换为大写:
List<String> names = List.of("Alice", "Bob", "Alex", "David"); List<String> result = names.stream() .filter(name -> name.startsWith("A")) .map(String::toUpperCase) .collect(Collectors.toList()); System.out.println(result); // Output: [ALICE, ALEX]
在此示例中:
java streams 为 java 带来了函数式编程功能,允许进行富有表现力和简洁的数据操作。通过了解中间操作和终端操作之间的区别以及如何有效地创建和使用流,您可以显着增强代码的可读性和可维护性。将流和 lambda 集成到您的工作流程中,以编写更干净、更高效的 java 应用程序。
祝直播愉快!
◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。