Stream流

Stream流

简洁优雅,可配合Lambda连用。Stream流把真正的函数式风格引入到Java中
举个例子:输出集合list中的开头为肘的长度为2的元素


public class StreamDemo {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<String>();
        list.add("肘子");
        list.add("红肠");
        list.add("玉米肠");
        list.add("冻豆腐");
        list.add("肘");
        list.add("肘");

        ArrayList<String> zhouList = new ArrayList<>();
        for (String s : list) {
            if (s.startsWith("肘")) {
                zhouList.add(s);
            }

        }
        ArrayList<String> threeList = new ArrayList<>();
        for (String s : zhouList) {
            if (s.length() == 2) {
                threeList.add(s);
            }
        }
        for (String s : threeList) {
            System.out.println(s);
        }
//		使用Stream流方式
//        list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length()==2).forEach(s -> System.out.println(s));
//		可直接阅读代码的字面意思,展示无关逻辑方式的语义:生成流、过滤开头为肘、过滤长度为2、逐一打印
        list.stream().filter(s -> s.startsWith("肘")).filter(s -> s.length() == 2).forEach(System.out::println);

    }
}
//输出结果
//肘子
//肘子

Stream流的使用

Stream流可分为生成流,中间操作,终结操作。

生成流

可通过数据源(集合、数组等)生成流。

Stream流的常见生成方式

  • Collection体系的集合可以使用默认方法stream​()生成流 default Stream stream​()
  • Map体系的集合间接的生成流
  • 数组可以通过Stream接口的静态方法of​(T…values)生成流

中间操作

一个流后面可以跟随零个或多个中间操作,其目的主要是打开流,做出某种程度的数据过滤/映射,然后返回一个新的流,交给下一个操作使用。都是Stream接口中的方法。

Stream流中的常见的中间操作方法

  • Stream filter​(Predicate predicate):用于对流中的数据进行过滤
    Predicate接口中的方法 boolean test​(T t):对给定的参数进行判断,返回一个布尔值
  • Stream limit​(long maxSize):返回此流中的元素组成的流,截取前指定参数个数的数据
  • Stream skip​(long n):跳过指定参数个数的数据,返回由该流的剩余元素组成的流
  • static Stream concat​(Stream a, Stream b):合并a和b两个流为一个流
  • Stream distinct​():返回由该流的不同元素(根据Object.equals(Object) )组成的流
  • Stream sorted​():返回由此流的元素组成的流,根据自然顺序排序
  • Stream sorted​(Comparator comparator):返回由该流的元素组成的流,根据提供的Comparator进行排序
  • Stream map​(Function mapper):返回由给定函数应用于此流的元素的结果组成的流
    Function接口中的方法 R apply​(T t)
  • IntStream mapToInt​(ToIntFunction mapper):返回一个IntStream其中包含将给定函数应用于此流的元素的结果
    IntStream:表示原始 int 流
    ToIntFunction接口中的方法 int applyAsInt​(T value)
    小例子:
public class Actor {
    String name;
    public Actor(String name) {
        this.name = name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
}
public class ActorDemo {
    public static void main(String[] args) {
        ArrayList<String> manList = new ArrayList<String>();
        manList.add("小雷");
        manList.add("陈小凡");
        manList.add("朱小磊");
        manList.add("李小白");
        manList.add("王小易");
        ArrayList<String> womenList = new ArrayList<String>();
        womenList.add("小敏");
        womenList.add("林清");
        womenList.add("林红");
        womenList.add("林数");
        Stream<String> man = manList.stream().filter(s -> s.length() == 3).limit(3);
        Stream<String> women = womenList.stream().filter(s -> s.startsWith("林")).skip(1);
        Stream<String> concat = Stream.concat(man, women);
        concat.map(Actor::new).forEach(s -> System.out.println(s.getName()));

    }
}
//输出结果
//陈小凡
//朱小磊
//李小白
//林红
//林数
public class InterStreamDemo {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<String>();
        list.add("spring");
        list.add("sum");
        list.add("winter");
        list.add("autumn");
        list.add("autumn");
        Stream<String> first = list.stream().limit(2).filter(s -> s.startsWith("s"));
        Stream<String> last = list.stream().skip(3);
//        Stream.concat(first,last).sorted().forEach(System.out::println);
        Stream<String> concat = Stream.concat(first, last);
        concat.sorted((s1,s2) -> {
            int num = s1.length()-s2.length();
            int num2 = num == 0? s1.compareTo(s2):num;
            return num2;
        }).distinct().forEach(System.out::println);


    }
}
//输出结果
//sum
//autumn
//spring

终结操作方法

一个流只能有一个终结操作,当这个操作执行后,流就被使用“光”了,无法再被操作。所以这必定是流的最后一个操作

Stream流的常见终结操作方法

  • void forEach​(Consumer action):对此流的每个元素执行操作
    Consumer接口中的方法 void accept​(T t):对给定的参数执行此操作
  • long count​():返回此流中的元素数
    小例子:
public class MapStreamDemo {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<String>();
        list.add("123");
        list.add("1");
        list.add("34");
        list.add("45");
        list.add("21");
        list.stream().map(Integer::parseInt).forEach(System.out::println);
        System.out.println("----------------");
        int sum = list.stream().mapToInt(Integer::parseInt).sum();
        System.out.println(sum);

    }
}
//输出结果
//123
//1
//34
//45
//21
//----------------
//224

Stream流的收集操作

对数据使用Stream流的方式操作完毕后,把流中的数据收集到集合中,可有一下的方法。

Stream流的收集方法

  • R collect​(Collector collector)
  • 但是这个收集方法的参数是一个Collector 接口

工具类Collectors提供了具体的收集方式

  • public static Collector toList​():把元素收集到List集合中
  • public static Collector toSet​():把元素收集到Set集合中
  • public static Collector toMap​(Function keyMapper,Function valueMapper):把元素收集到Map集合中
public class CollectionStreamDemo {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<String>();
        list.add("春天");
        list.add("煎饼果子");
        list.add("小笼包");
        list.add("地三鲜");
        Stream<String> stringStream = list.stream().filter(s -> s.length() > 2);
        List<String> collect = stringStream.collect(Collectors.toList());
        for (String food : collect) {
            System.out.println(food);
        }
	}
}
//输出结果
//煎饼果子
//小笼包
//地三鲜
		Set<Integer> set = new HashSet<Integer>();
        set.add(10);
        set.add(10);
        set.add(20);
        set.add(34);
        set.add(51);

        System.out.println("------------");
        Stream<Integer> stream1 = set.stream().filter(s -> s > 25);
        Set<Integer> collect1 = stream1.collect(Collectors.toSet());
        for (Integer collect11 : collect1) {
            System.out.println(collect11);
        }
        
//输出结果
//34
//51
		String[] strAttay = {"小张,34","小李,31","小数,41","小巩,56","小薇,24"};
        Stream<String> stringStream1 = Stream.of(strAttay).filter(s -> Integer.parseInt(s.split(",")[1]) > 28);
        Map<String, Integer> collect2 = stringStream1.collect(Collectors.toMap(s -> s.split(",")[0], s -> Integer.parseInt(s.split(",")[1])));
        Set<String> keySet = collect2.keySet();
        for (String keset : keySet) {
            Integer integer = collect2.get(keset);
            
            System.out.println(keset+","+integer);
        }
//输出结果        
//小巩,56
//小李,31
//小张,34
//小数,41	

        
<think>好的,用户想要找关于Java Stream的练习题和学习资料。首先,我需要回忆一下相关的知识点,并结合提供的引用内容来整理答案。 首先,用户提供的引用中有四个关于Stream的资料,其中提到了练习题、示例代码以及基础教程。比如引用3提到StreamJava 8的重要特性,允许声明式处理集合,这可能适合作为教程部分的推荐。引用1和引用2给出了具体的练习题,如转换并行、处理绝对值、使用flatMap拆分单词等。引用4则展示了过滤偶数的例子,这些都是很好的练习题素材。 接下来,我需要将这些内容分类整理,可能分为基础概念、练习题推荐和学习资源。同时,用户可能需要教程链接或书籍推荐,但根据引用中没有提供外部链接,可能需要建议官方文档或知名网站。 还要注意用户提到的系统级指令,比如数学公式的格式,但这次问题不涉及数学内容,所以可能不需要处理这部分。另外,生成相关问题的时候,要确保问题与Stream相关,并且基于已有的引用内容。例如,如何优化Stream性能、并行的使用场景、map和filter的区别等。 需要检查是否有遗漏的要点,比如用户可能对Stream的中间操作和终端操作有疑问,或者想了解常见的陷阱。所以在相关问题里可以包括这些方面。 最后,确保回答结构清晰,分点列出,并且每个部分都有对应的引用支持。例如,在练习题部分具体列出每个练习并标注引用来源,这样用户能清楚知道每个例子出自哪里。同时,学习资源部分可以推荐官方文档和书籍,如《Java 8实战》。 总结一下,回答应该包括基础概念、练习题示例、学习资源和相关问题,确保每个部分都有引用支持,并且语言自然畅。</think>### Java Stream学习资源与练习题整理 #### 一、Stream基础概念 StreamJava 8的核心特性之一,允许以声明性方式处理集合数据,支持函数式编程风格。例如: ```java list.stream().filter(n -> n % 2 == 0).forEach(System.out::println); ``` 这段代码通过`filter`筛选偶数并输出[^4]。Stream的核心操作包括中间操作(如`map`、`filter`)和终端操作(如`forEach`、`collect`)[^3]。 --- #### 二、经典练习题推荐(含代码示例) 1. **取元素绝对值** ```java Stream.of(1, -2, -3, 4, -5) .map(Math::abs) .forEach(System.out::println); // 输出1,2,3,4,5 [^2] ``` 2. **拆分单词并合并** ```java List<String> words = Arrays.asList("Hello World", "Java Stream"); words.stream() .flatMap(word -> Arrays.stream(word.split(" "))) .forEach(System.out::println); // 输出拆分后的单词[^1] ``` 3. **过滤奇数保留偶数** ```java List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10); list.stream() .filter(n -> n % 2 == 0) .forEach(System.out::println); // 输出2,4,6,8,10 ``` 4. **并行加速计算** ```java list.parallelStream() // 转换为并行 .filter(...); // 利用多核处理器提升性能[^1] ``` --- #### 三、学习资源推荐 1. **官方文档** Oracle Java 8 Stream官方教程:系统介绍Stream API的设计与使用场景。 2. **书籍《Java 8实战》** 详细讲解Stream、Lambda表达式等特性,含大量代码示例。 3. **综合教程与练习题集** - 的入门教程:涵盖Stream操作分类(中间操作、终端操作)和常见用例。 - 中的练习题:覆盖`map`、`filter`、`flatMap`和并行等核心操作。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值