Stream流的中间方法详解
Stream流的中间方法指那些返回新Stream的操作,允许链式调用。这些方法通常用于数据过滤、映射、排序等操作,不会触发最终计算。
filter方法
filter用于筛选满足条件的元素,接受一个Predicate函数式接口参数。
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
List<Integer> evenNumbers = numbers.stream().filter(n -> n % 2 == 0).collect(Collectors.toList());
// 输出: [2,4]
map方法
map用于元素转换,接受Function接口参数,将元素映射为另一种形式。
List<String> names = Arrays.asList("Alice", "Bob");
List<Integer> nameLengths = names.stream().map(String::length).collect(Collectors.toList());
// 输出: [5,3]
flatMap方法
flatMap用于将嵌套结构扁平化,每个元素转换为新的Stream后合并。
List<List<Integer>> nestedNumbers = Arrays.asList(Arrays.asList(1,2),Arrays.asList(3,4));
List<Integer> flattened = nestedNumbers.stream().flatMap(List::stream).collect(Collectors.toList());
// 输出: [1,2,3,4]
distinct方法
distinct去除重复元素,依赖元素的equals方法。
List<Integer> duplicates = Arrays.asList(1,2,2,3);
List<Integer> unique = duplicates.stream().distinct().collect(Collectors.toList());
// 输出: [1,2,3]
sorted方法
sorted用于排序,可自然排序或提供Comparator。
List<String> unsorted = Arrays.asList("Bob", "Alice");
List<String> sorted = unsorted.stream().sorted().collect(Collectors.toList());
// 输出: ["Alice","Bob"]List<Integer> numbers = Arrays.asList(3,1,2);
List<Integer> customSorted = numbers.stream().sorted(Comparator.reverseOrder()).collect(Collectors.toList());
// 输出: [3,2,1]
limit方法
limit截取前N个元素。
List<Integer> numbers = Arrays.asList(1,2,3,4,5);
List<Integer> firstThree = numbers.stream().limit(3).collect(Collectors.toList());
// 输出: [1,2,3]
skip方法
skip跳过前N个元素。
List<Integer> numbers = Arrays.asList(1,2,3,4,5);
List<Integer> afterTwo = numbers.stream().skip(2).collect(Collectors.toList());
// 输出: [3,4,5]
peek方法
peek用于调试,允许查看流经的元素但不修改。
List<String> names = Arrays.asList("Alice", "Bob");
List<String> result = names.stream().peek(System.out::println).map(String::toUpperCase).collect(Collectors.toList());
// 输出调试信息: Alice Bob
// 最终结果: ["ALICE","BOB"]
实例一:
public class StreamDemo6 {public static void main(String[] args) {/*filter 过滤limit 获取前几个元素skip 跳过前几个元素注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据*/ArrayList<String> list = new ArrayList<>();Collections.addAll(list, "张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤");//filter 过滤 把张开头的留下,其余数据过滤不要/*list.stream().filter(new Predicate<String>() {@Overridepublic boolean test(String s) {//如果返回值为true,表示当前数据留下//如果返回值为false,表示当前数据舍弃不要return s.startsWith("张");}}).forEach(s -> System.out.println(s));list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length() == 3).forEach(s -> System.out.println(s));System.out.println("====================================");System.out.println(list);*//* limit 获取前几个元素skip 跳过前几个元素*///"张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤"//list.stream().limit(3).forEach(s -> System.out.println(s));//list.stream().skip(4) .forEach(s -> System.out.println(s));//课堂练习://"张强", "张三丰", "张翠山"//第一种思路://先获取前面6个元素:"张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山",//然后跳过前面3个元素//list.stream().limit(6).skip(3).forEach(s -> System.out.println(s));//第二种思路://先跳过3个元素:"张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤"//然后再获取前面3个元素:"张强", "张三丰", "张翠山"//list.stream().skip(3).limit(3).forEach(s -> System.out.println(s));}
}
实例二:
public class StreamDemo7 {public static void main(String[] args) {/*distinct 元素去重,依赖(hashCode和equals方法)concat 合并a和b两个流为一个流注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据*/ArrayList<String> list1 = new ArrayList<>();Collections.addAll(list1, "张无忌","张无忌","张无忌", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤");ArrayList<String> list2 = new ArrayList<>();Collections.addAll(list2, "周芷若", "赵敏");// distinct 元素去重,依赖(hashCode和equals方法)//list1.stream().distinct().forEach(s -> System.out.println(s));Stream.concat(list1.stream(),list2.stream()).forEach(s -> System.out.println(s));}
}
实例三:
public class StreamDemo8 {public static void main(String[] args) {/*map 转换流中的数据类型注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据*/ArrayList<String> list = new ArrayList<>();Collections.addAll(list, "张无忌-15", "周芷若-14", "赵敏-13", "张强-20", "张三丰-100", "张翠山-40", "张良-35", "王二麻子-37", "谢广坤-41");//需求:只获取里面的年龄并进行打印//String->int//第一个类型:流中原本的数据类型//第二个类型:要转成之后的类型//apply的形参s:依次表示流里面的每一个数据//返回值:表示转换之后的数据//当map方法执行完毕之后,流上的数据就变成了整数//所以在下面forEach当中,s依次表示流里面的每一个数据,这个数据现在就是整数了list.stream().map(new Function<String, Integer>() {@Overridepublic Integer apply(String s) {String[] arr = s.split("-");String ageString = arr[1];int age = Integer.parseInt(ageString);return age;}}).forEach(s-> System.out.println(s));System.out.println("------------------------");list.stream().map(s-> Integer.parseInt(s.split("-")[1])).forEach(s-> System.out.println(s));}
}
注意:中间操作具有延迟特性,只有遇到终端操作时才会执行。多个中间操作可以组合形成处理流水线。