美文网首页
java8之二——使用流

java8之二——使用流

作者: 康康不遛猫 | 来源:发表于2017-08-26 23:43 被阅读0次

二、流处理
1、流的概念
流:从支持数据操作的源,生产而来的元素序列。
这里源可以是集合,输入/数据流等,有序序列生成的元素序列仍保持之前的顺序。
流操作分为中间操作和终端操作,可以顺序执行也可以并行执行。
流操作特点:流水线、内部迭代(相比之前为外部迭代或显式迭代)。


图片.png

注意:终端操作只能进行一次,如不能forEach后再次forEach

public static List<String> getLowCaloricDishesNamesInJava8(List<Dish> dishes){
        return dishes.stream()
                .filter(d -> d.getCalories() < 400)
                .sorted(comparing(Dish::getCalories))
                .map(Dish::getName)
                .collect(toList());
    }

流的性能问题:http://www.importnew.com/17262.html

2、 常用流操作
(1)筛选切片
Streams接口支持filter方法,该接收接收一个谓词(返回boolean的函数)作为参数,返回一个符合谓词的元素的流。如:

 // Filtering unique elements
        List<Integer> numbers = Arrays.asList(1, 2, 1, 3, 3, 2, 4);
        numbers.stream()
               .filter(i -> i % 2 == 0)
               .distinct()//对元素做去重
               .forEach(System.out::println);

        // Truncating a stream
        List<Dish> dishesLimit3 =
            menu.stream()
                .filter(d -> d.getCalories() > 300)
                .limit(3)//返回流的前3个
                .collect(toList());

        dishesLimit3.forEach(System.out::println);

        // Skipping elements
        List<Dish> dishesSkip2 =
            menu.stream()
                .filter(d -> d.getCalories() > 300)
                .skip(2)//返回流的后2个
                .collect(toList());

        dishesSkip2.forEach(System.out::println);

(2)映射:map与flatmap
StreamAPI的map方法、flatMap对流中每个元素应用函数,不同之处在于flatMap将流中每个元素转化一个流,然后把所有流连成一个流(扁平化)。

 // map
        List<String> dishNames = menu.stream()
                                     .map(Dish::getName)
                                     .collect(toList());
        System.out.println(dishNames);

        // map
        List<String> words = Arrays.asList("Hello", "World");
        List<Integer> wordLengths = words.stream()
                                         .map(String::length)
                                         .collect(toList());//通过collect将流Stream转成集合
        System.out.println(wordLengths);

        // flatMap
        words.stream()
                 .flatMap((String line) -> Arrays.stream(line.split("")))
                 .distinct()
                 .forEach(System.out::println);

        // flatMap
        List<Integer> numbers1 = Arrays.asList(1,2,3,4,5);
        List<Integer> numbers2 = Arrays.asList(6,7,8);
        List<int[]> pairs =
                        numbers1.stream()
                                .flatMap((Integer i) -> numbers2.stream()
                                                       .map((Integer j) -> new int[]{i, j})
                                 )
                                .filter(pair -> (pair[0] + pair[1]) % 3 == 0)
                                .collect(toList());
        pairs.forEach(pair -> System.out.println("(" + pair[0] + ", " + pair[1] + ")"));
        //输出(1,8)、(2,7)、(3,6)
Paste_Image.png Paste_Image.png

(3)查找和匹配
StreamAPI提供方法:
allMatch:检查谓词是否匹配所有元素
anyMatch:检查谓词是否至少匹配一个元素
noneMatch:相对于allMatch,检查流中没有任何元素和谓词匹配

findAny:返回流中任意元素
findFirst:查找流中第一个元素

public static void main(String...args){
        if(isVegetarianFriendlyMenu()){
            System.out.println("Vegetarian friendly");
        }

        System.out.println(isHealthyMenu());
        System.out.println(isHealthyMenu2());
        
        Optional<Dish> dish = findVegetarianDish();
        dish.ifPresent(d -> System.out.println(d.getName()));
    }
    
    private static boolean isVegetarianFriendlyMenu(){
        return menu.stream().anyMatch(Dish::isVegetarian);
    }
    
    private static boolean isHealthyMenu(){
        return menu.stream().allMatch(d -> d.getCalories() < 1000);
    }
    
    private static boolean isHealthyMenu2(){
        return menu.stream().noneMatch(d -> d.getCalories() >= 1000);
    }
    
   //findAny返回一个Optional容器实例,找不到元素避免使用null
   //isPresent :Optional有值返回true
   //ifPresent:有值时执行其中代码块
    private static Optional<Dish> findVegetarianDish(){
        return menu.stream().filter(Dish::isVegetarian).findAny();
    }

(4)规约
reduce操作将流中所有元素反复结合。

public static void main(String...args){

        List<Integer> numbers = Arrays.asList(3,4,5,1,2);
        int sum = numbers.stream().reduce(0, (a, b) -> a + b);//求和
        System.out.println(sum);

        int sum2 = numbers.stream().reduce(0, Integer::sum);
        System.out.println(sum2);

        int max = numbers.stream().reduce(0, (a, b) -> Integer.max(a, b));
        System.out.println(max);

        Optional<Integer> min = numbers.stream().reduce(Integer::min);
        min.ifPresent(System.out::println);
        //没有初始值,流为空则没有返回,因此返回使用Optional

        int calories = menu.stream()
                           .map(Dish::getCalories)
                           .reduce(0, Integer::sum);
        System.out.println("Number of calories:" + calories);
    }
Paste_Image.png

(5)数值流:
java8引入三个原始类型特化流IntStream、DoubleStream、LongSteam,分别将流中元素特化为int、long、double。
boxed()方法则对原始类型进行装箱。
数值流自带sum、max、min、average等方法。

int calories = menu.stream()
                           .mapToInt(Dish::getCalories)
                           .sum();
        System.out.println("Number of calories:" + calories);


        // max and OptionalInt
        OptionalInt maxCalories = menu.stream()                                                      
                                      .mapToInt(Dish::getCalories)
                                      .max();

        int max = maxCalories.orElse(1);//没有最大值,设置默认值1

        // numeric ranges
        IntStream evenNumbers = IntStream.rangeClosed(1, 100)
                                 .filter(n -> n % 2 == 0);
       //rangeClosed包含结束值,而range不包含结束值

        System.out.println(evenNumbers.count());
        
        //返回(1,100)符合勾股定理的三元组
        Stream<int[]> pythagoreanTriples =
               IntStream.rangeClosed(1, 100).boxed()
                        .flatMap(a -> IntStream.rangeClosed(a, 100)
                                               .mapToObj(b -> new int[]{a, b, (int) Math.sqrt(a * a + b * b)})
                                               .filter(t -> t[2] % 1 == 0)
                                               );       
        pythagoreanTriples.forEach(t -> System.out.println(t[0] + ", " + t[1] + ", " + t[2])); 

(6)构建流
流可以由值、数组、文件创建

        // Stream.of
        Stream<String> stream = Stream.of("Java 8", "Lambdas", "In", "Action");
        stream.map(String::toUpperCase).forEach(System.out::println);

        // Stream.empty
        Stream<String> emptyStream = Stream.empty();

        // Arrays.stream
        int[] numbers = {2, 3, 5, 7, 11, 13};
        System.out.println(Arrays.stream(numbers).sum());
       
       //文件流回自动关闭
       long uniqueWords = Files.lines(Paths.get("lambdasinaction/chap5/data.txt"), Charset.defaultCharset())
                                 .flatMap(line -> Arrays.stream(line.split(" ")))//生成单词流
                                 .distinct()
                                 .count();//

         System.out.println("There are " + uniqueWords + " unique words in data.txt");

可以由于Stream.iterate和Stream.generate函数生成无限流(没有固定大小)

        // Stream.iterate
        Stream.iterate(0, n -> n + 2)
              .limit(10)
              .forEach(System.out::println);

        // fibonnaci with iterate
        Stream.iterate(new int[]{0, 1}, t -> new int[]{t[1],t[0] + t[1]})
              .limit(10)
              .forEach(t -> System.out.println("(" + t[0] + ", " + t[1] + ")"));
        
        Stream.iterate(new int[]{0, 1}, t -> new int[]{t[1],t[0] + t[1]})
              .limit(10)
              . map(t -> t[0])  
              .forEach(System.out::println);

        // random stream of doubles with Stream.generate
        Stream.generate(Math::random)
              .limit(10)
              .forEach(System.out::println);
 
        // stream of 1s with Stream.generate
        IntStream.generate(() -> 1)
                 .limit(5)
                 .forEach(System.out::println);

        IntStream.generate(new IntSupplier(){
            public int getAsInt(){
                return 2;
            }
        }).limit(5)
          .forEach(System.out::println);

相关文章

网友评论

      本文标题:java8之二——使用流

      本文链接:https://www.haomeiwen.com/subject/qgevrxtx.html