Java 8 API添加了一个新的抽象称为流Stream,可让你以一种声明的方式处理数据。java
Stream 使用一种相似用 SQL 语句从数据库查询数据的直观方式来提供一种对 Java 集合运算和表达的高阶抽象。程序员
Stream API能够极大提升Java程序员的生产力,让程序员写出高效率、干净、简洁的代码。数据库
这种风格将要处理的元素集合看做一种流, 流在管道中传输, 而且能够在管道的节点上进行处理, 好比筛选, 排序,聚合等。数组
元素流在管道中通过中间操做(intermediate operation)的处理,最后由最终操做(terminal operation)获得前面处理的结果。bash
+--------------------+ +------+ +------+ +---+ +-------+
| stream of elements +-----> |filter+-> |sorted+-> |map+-> |collect|
+--------------------+ +------+ +------+ +---+ +-------+复制代码
以上的流程转换为 Java 代码为:app
List<Integer> transactionsIds =
widgets.stream()
.filter(b -> b.getColor() == RED)
.sorted((x,y) -> x.getWeight() - y.getWeight())
.mapToInt(Widget::getWeight)
.sum();复制代码
Stream(流)是一个来自数据源的元素队列并支持聚合操做dom
和之前的Collection操做不一样, Stream操做还有两个基础的特征:优化
在 Java 8 中, 集合接口有两个方法来生成流:ui
List<String> strings = Arrays.asList(“abc“, ““, “bc“, “efg“, “abcd“,““, “jkl“); List<String> filtered = strings.stream().filter(string -> !string.isEmpty()).collect(Collectors.toList());
Stream 提供了新的方法 ‘forEach’ 来迭代流中的每一个数据。如下代码片断使用 forEach 输出了10个随机数:spa
Random random = new Random(); random.ints().limit(10).forEach(System.out::println);
map 方法用于映射每一个元素到对应的结果,如下代码片断使用 map 输出了元素对应的平方数:
List<Integer> numbers = Arrays.asList(3, 2, 2, 3, 7, 3, 5); // 获取对应的平方数 List<Integer> squaresList = numbers.stream().map( i -> i*i).distinct().collect(Collectors.toList());
filter 方法用于经过设置的条件过滤出元素。如下代码片断使用 filter 方法过滤出空字符串:
List<String>strings = Arrays.asList(“abc“, ““, “bc“, “efg“, “abcd“,““, “jkl“); // 获取空字符串的数量 int count= strings.stream().filter(string -> string.isEmpty()).count();
limit 方法用于获取指定数量的流。 如下代码片断使用 limit 方法打印出 10 条数据:
Random random = new Random(); random.ints().limit(10).forEach(System.out::println);
sorted 方法用于对流进行排序。如下代码片断使用 sorted 方法对输出的 10 个随机数进行排序:
Random random = new Random(); random.ints().limit(10).sorted().forEach(System.out::println);
parallelStream 是流并行处理程序的代替方法。如下实例咱们使用 parallelStream 来输出空字符串的数量:
List<String> strings = Arrays.asList(“abc“, ““, “bc“, “efg“, “abcd“,““, “jkl“); // 获取空字符串的数量 intcount = strings.parallelStream().filter(string -> string.isEmpty()).count();
咱们能够很容易的在顺序运行和并行直接切换。
Collectors 类实现了不少归约操做,例如将流转换成集合和聚合元素。Collectors 可用于返回列表或字符串:
List<String>strings = Arrays.asList(“abc“, ““, “bc“, “efg“, “abcd“,““, “jkl“);List<String> filtered = strings.stream().filter(string -> !string.isEmpty()).collect(Collectors.toList()); System.out.println(“筛选列表: “ + filtered);String mergedString = strings.stream().filter(string -> !string.isEmpty()).collect(Collectors.joining(“, “)); System.out.println(“合并字符串: “ + mergedString);
另外,一些产生统计结果的收集器也很是有用。它们主要用于int、double、long等基本类型上,它们能够用来产生相似以下的统计结果。
List<Integer> numbers = Arrays.asList(3, 2, 2, 3, 7, 3, 5);IntSummaryStatistics stats = integers.stream().mapToInt((x) -> x).summaryStatistics();System.out.println(“列表中最大的数 : “ + stats.getMax());System.out.println(“列表中最小的数 : “ + stats.getMin());System.out.println(“全部数之和 : “ + stats.getSum());System.out.println(“平均数 : “ + stats.getAverage());
将如下代码放入 Java8Tester.java 文件中:
package org.java.base.stream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.IntSummaryStatistics;
import java.util.List;
import java.util.Random;
import java.util.stream.Collectors;
import java.util.Map;public class Java8Tester {
public static void main(String args[]){
System.out.println(“使用 Java 7: “);// 计算空字符串
List<String> strings = Arrays.asList(“abc”, “”, “bc”, “efg”, “abcd”,””, “jkl”);
System.out.println(“列表: ” +strings);
long count = getCountEmptyStringUsingJava7(strings);System.out.println(“空字符数量为: ” + count);
count = getCountLength3UsingJava7(strings);System.out.println(“字符串长度为 3 的数量为: ” + count);
// 删除空字符串
List<String> filtered = deleteEmptyStringsUsingJava7(strings);
System.out.println(“筛选后的列表: ” + filtered);// 删除空字符串,并使用逗号把它们合并起来
String mergedString = getMergedStringUsingJava7(strings,”, “);
System.out.println(“合并字符串: ” + mergedString);
List<Integer> numbers = Arrays.asList(3, 2, 2, 3, 7, 3, 5);// 获取列表元素平方数
List<Integer> squaresList = getSquares(numbers);
System.out.println(“平方数列表: ” + squaresList);
List<Integer> integers = Arrays.asList(1,2,13,4,15,6,17,8,19);System.out.println(“列表: ” +integers);
System.out.println(“列表中最大的数 : ” + getMax(integers));
System.out.println(“列表中最小的数 : ” + getMin(integers));
System.out.println(“全部数之和 : ” + getSum(integers));
System.out.println(“平均数 : ” + getAverage(integers));
System.out.println(“随机数: “);// 输出10个随机数
Random random = new Random();for(int i=0; i < 10; i++){
System.out.println(random.nextInt());
}System.out.println(“使用 Java 8: “);
System.out.println(“列表: ” +strings);count = strings.stream().filter(string->string.isEmpty()).count();
System.out.println(“空字符串数量为: ” + count);count = strings.stream().filter(string -> string.length() == 3).count();
System.out.println(“字符串长度为 3 的数量为: ” + count);filtered = strings.stream().filter(string ->!string.isEmpty()).collect(Collectors.toList());
System.out.println(“筛选后的列表: ” + filtered);mergedString = strings.stream().filter(string ->!string.isEmpty()).collect(Collectors.joining(“, “));
System.out.println(“合并字符串: ” + mergedString);squaresList = numbers.stream().map( i ->i*i).distinct().collect(Collectors.toList());
System.out.println(“Squares List: ” + squaresList);
System.out.println(“列表: ” +integers);IntSummaryStatistics stats = integers.stream().mapToInt((x) ->x).summaryStatistics();
System.out.println(“列表中最大的数 : ” + stats.getMax());
System.out.println(“列表中最小的数 : ” + stats.getMin());
System.out.println(“全部数之和 : ” + stats.getSum());
System.out.println(“平均数 : ” + stats.getAverage());
System.out.println(“随机数: “);random.ints().limit(10).sorted().forEach(System.out::println);
// 并行处理
count = strings.parallelStream().filter(string -> string.isEmpty()).count();
System.out.println(“空字符串的数量为: ” + count);
}private static int getCountEmptyStringUsingJava7(List<String> strings){
int count = 0;for(String string: strings){
if(string.isEmpty()){
count++;
}
}
return count;
}private static int getCountLength3UsingJava7(List<String> strings){
int count = 0;for(String string: strings){
if(string.length() == 3){
count++;
}
}
return count;
}private static List<String> deleteEmptyStringsUsingJava7(List<String> strings){
List<String> filteredList = new ArrayList<String>();for(String string: strings){
if(!string.isEmpty()){
filteredList.add(string);
}
}
return filteredList;
}private static String getMergedStringUsingJava7(List<String> strings, String separator){
StringBuilder stringBuilder = new StringBuilder();for(String string: strings){
if(!string.isEmpty()){
stringBuilder.append(string);
stringBuilder.append(separator);
}
}
String mergedString = stringBuilder.toString();
return mergedString.substring(0, mergedString.length()-2);
}private static List<Integer> getSquares(List<Integer> numbers){
List<Integer> squaresList = new ArrayList<Integer>();for(Integer number: numbers){
Integer square = new Integer(number.intValue() * number.intValue());if(!squaresList.contains(square)){
squaresList.add(square);
}
}
return squaresList;
}private static int getMax(List<Integer> numbers){
int max = numbers.get(0);for(int i=1;i < numbers.size();i++){
Integer number = numbers.get(i);
if(number.intValue() > max){
max = number.intValue();
}
}
return max;
}private static int getMin(List<Integer> numbers){
int min = numbers.get(0);for(int i=1;i < numbers.size();i++){
Integer number = numbers.get(i);if(number.intValue() < min){
min = number.intValue();
}
}
return min;
}private static int getSum(List numbers){
int sum = (int)(numbers.get(0));for(int i=1;i < numbers.size();i++){
sum += (int)numbers.get(i);
}
return sum;
}private static int getAverage(List<Integer> numbers){
return getSum(numbers) / numbers.size();
}
}
执行以上脚本,输出结果为:
$ javac Java8Tester.java
$ java Java8Tester
使用 Java 7:
列表: [abc, , bc, efg, abcd, , jkl]
空字符数量为: 2
字符串长度为 3 的数量为: 3
筛选后的列表: [abc, bc, efg, abcd, jkl]
合并字符串: abc, bc, efg, abcd, jkl
平方数列表: [9, 4, 49, 25]
列表: [1, 2, 13, 4, 15, 6, 17, 8, 19]
列表中最大的数 : 19
列表中最小的数 : 1
全部数之和 : 85
平均数 : 9
随机数:
-393170844
-963842252
447036679
-1043163142
-881079698
221586850
-1101570113
576190039
-1045184578
1647841045
使用 Java 8:
列表: [abc, , bc, efg, abcd, , jkl]
空字符串数量为: 2
字符串长度为 3 的数量为: 3
筛选后的列表: [abc, bc, efg, abcd, jkl]
合并字符串: abc, bc, efg, abcd, jkl
Squares List: [9, 4, 49, 25]
列表: [1, 2, 13, 4, 15, 6, 17, 8, 19]
列表中最大的数 : 19
列表中最小的数 : 1
全部数之和 : 85
平均数 : 9.444444444444445
随机数:
-1743813696
-1301974944
-1299484995
-779981186
136544902
555792023
1243315896
1264920849
1472077135
1706423674
空字符串的数量为: 2复制代码