当前位置 博文首页 > 白河寒秋XY的博客:JAVA从零开始学习知识整理——day13—【Strea

    白河寒秋XY的博客:JAVA从零开始学习知识整理——day13—【Strea

    作者:[db:作者] 时间:2021-08-04 15:06

    一、Stream流
    1.1 Stream流引入
    传统的数据筛选等操作,需要根据筛选条件进行多次操作,除了典型的必需的添加、删除、获取外,其中最为常见的就是通过遍历集合等。
    例如传统的方法:
    三个循环完成三个筛选条件:

    import java.util.ArrayList;
    import java.util.List;
    public class Demo02NormalFilter {
    		public static void main(String[] args) {
    				List<String> list = new ArrayList<>();
    				list.add("张无忌");
    				list.add("周芷若");
    				list.add("赵敏");
    				list.add("张强");
    				list.add("张三丰");
    				List<String> zhangList = new ArrayList<>();
    				for (String name : list) {
    						if (name.startsWith("张")) {
    							zhangList.add(name);
    						}
    				}
    				List<String> shortList = new ArrayList<>();
    				for (String name : zhangList) {
    				if (name.length() == 3) {
    							shortList.add(name);
    						}
    				}
    				for (String name : shortList) {
    				System.out.println(name);
    			}
    		}
    }
    

    Stream的更优写法:

    import java.util.ArrayList;
    import java.util.List;
    public class Demo03StreamFilter {
    		public static void main(String[] args) {
    			List<String> list = new ArrayList<>();
    			list.add("张无忌");
    			list.add("周芷若");
    			list.add("赵敏");
    			list.add("张强");
    			list.add("张三丰");
    			list.stream()
    								.filter(s ‐> s.startsWith("张"))
    								.filter(s ‐> s.length() == 3)
    								.forEach(System.out::println);
    			}
    }
    

    1.2 流式思想概述
    在上面的例子里,过滤遍历等操作其实都是对流对象的处理,集合对象没有被真的处理,只有终结方法执行的时候,集合才会被按流程处理,这得益于lambda的延时执行特征。

    **Stream(流)**是一个来自数据源的元素队列元素是特定类型的对象,形成一个队列。 Java中的Stream并不会存储元素,而是按需计算。
    数据源 流的来源。 可以是集合,数组 等。
    和以前的Collection操作不同, Stream操作还有两个基础的特征:
    Pipelining: 中间操作都会返回流对象本身。 这样多个操作可以串联成一个管道, 如同流式风格(fluent
    style)。 这样做可以对操作进行优化, 比如延迟执行(laziness)和短路( short-circuiting)。
    内部迭代: 以前对集合遍历都是通过Iterator或者增强for的方式, 显式的在集合外部进行迭代, 这叫做外部迭
    代。 Stream提供了内部迭代的方式,流可以直接调用遍历方法。

    1.3获取流
    java.util.stream.Stream 是Java 8新加入的最常用的流接口。(这并不是一个函数式接口。)
    获取一个流非常简单,有以下几种常用的方式:

    所有的 Collection 集合都可以通过 stream 默认方法获取流;
    Stream 接口的静态方法 of 可以获取数组对应的流。

    1.3.1 根据Collection获取流
    首先, java.util.Collection 接口中加入了default方法 stream 用来获取流,所以其所有实现类均可获取流。

    import java.util.*;
    import java.util.stream.Stream;
    public class Demo04GetStream {
    		public static void main(String[] args) {
    				List<String> list = new ArrayList<>();
    				// ...
    				Stream<String> stream1 = list.stream();
    				Set<String> set = new HashSet<>();
    				// ...
    				Stream<String> stream2 = set.stream();
    				Vector<String> vector = new Vector<>();
    				// ...
    				Stream<String> stream3 = vector.stream();
    		}
    }
    

    1.3.2 根据Map获取流
    java.util.Map 接口不是 Collection 的子接口,且其K-V数据结构不符合流元素的单一特征,所以获取对应的流
    需要分key、value或entry等情况:

    import java.util.HashMap;
    import java.util.Map;
    import java.util.stream.Stream;
    public class Demo05GetStream {
    		public static void main(String[] args) {
    				Map<String, String> map = new HashMap<>();
    				// ...
    				Stream<String> keyStream = map.keySet().stream();
    				Stream<String> valueStream = map.values().stream();
    				Stream<Map.Entry<String, String>> entryStream = map.entrySet().stream();
    		}
    }
    

    1.4 根据数组获取流
    如果使用的不是集合或映射而是数组,由于数组对象不可能添加默认方法,所以 Stream 接口中提供了静态方法of ,使用很简单:

    import java.util.stream.Stream;
    public class Demo06GetStream {
    public static void main(String[] args) {
    			String[] array = { "张无忌", "张翠山", "张三丰", "张一元" };
    			Stream<String> stream = Stream.of(array);
    	}
    }
    

    1.4 常用方法
    流模型的操作很丰富,这里介绍一些常用的API。这些方法可以被分成两种:
    延迟方法:返回值类型仍然是 Stream 接口自身类型的方法,因此支持链式调用。(除了终结方法外,其余方法均为延迟方法。)
    终结方法:返回值类型不再是 Stream 接口自身类型的方法,因此不再支持类似 StringBuilder 那样的链式调
    用。本小节中,终结方法包括 count 和 forEach 方法。
    逐一处理:forEach

    import java.util.stream.Stream;
    public class Demo12StreamForEach {
    		public static void main(String[] args) {
    			Stream<String> stream = Stream.of("张无忌", "张三丰", "周芷若");
    			stream.forEach(name‐> System.out.println(name));
    	}
    }
    

    过滤:filter

    import java.util.stream.Stream;
    public class Demo07StreamFilter {
    		public static void main(String[] args) {
    				Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
    				Stream<String> result = original.filter(s ‐> s.startsWith("张"));
    		}
    }
    

    映射:map

    import java.util.stream.Stream;
    public class Demo08StreamMap {
    		public static void main(String[] args) {
    			Stream<String> original = Stream.of("10", "12", "18");
    			Stream<Integer> result = original.map(str‐>Integer.parseInt(str));
    	}
    }
    

    统计个数:count

    import java.util.stream.Stream;
    public class Demo09StreamCount {
    		public static void main(String[] args) {
    			Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
    			Stream<String> result = original.filter(s ‐> s.startsWith("张"));
    			System.out.println(result.count()); // 2
    		}
    }
    

    取用前几个:limit

    import java.util.stream.Stream;
    public class Demo10StreamLimit {
    			public static void main(String[] args) {
    				Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
    				Stream<String> result = original.limit(2);
    				System.out.println(result.count()); // 2
    		}
    }
    

    跳过前几个:skip

    public class Demo11StreamSkip {
    	public static void main(String[] args) {
    			Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
    			Stream<String> result = original.skip(2);
    			System.out.println(result.count()); // 1
    	}
    }
    

    组合:concat

    import java.util.stream.Stream;
    public class Demo12StreamConcat {
    		public static void main(String[] args) {
    				Stream<String> streamA = Stream.of("张无忌");
    				Stream<String> streamB = Stream.of("张翠山");
    				Stream<String> result = Stream.concat(streamA, streamB);
    		}
    }
    

    实例:
    现在有两个 ArrayList 集合存储队伍当中的多个成员姓名,要求使用传统的for循环(或增强for循环)依次进行以
    下若干操作步骤:

    1. 第一个队伍只要名字为3个字的成员姓名;存储到一个新集合中。
    2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
    3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
    4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
    5. 将两个队伍合并为一个队伍;存储到一个新集合中。
    6. 根据姓名创建 Person 对象;存储到一个新集合中。
    7. 打印整个队伍的Person对象信息。
      代码实现:(Person类自建)
    import java.util.ArrayList;
    import java.util.stream.Stream;
    
    public class demo {
    
            public static void main(String[] args) {
    //第一支队伍
                ArrayList<String> one = new ArrayList<>();
                one.add("迪丽热巴");
                one.add("宋远桥");
                one.add("苏星河");
                one.add("石破天");
                one.add("石中玉");
                one.add("老子");
                one.add("庄子");
                one.add("洪七公");
    //第二支队伍
                ArrayList<String> two = new ArrayList<>();
                two.add("古力娜扎");
                two.add("张无忌");
                two.add("赵丽颖");
                two.add("张三丰");
                two.add("尼古拉斯赵四");
                two.add("张天爱");
                two.add("张二狗");
    // ....
                Stream<String> stream = one.stream();
                Stream<String> limit = stream.filter((name) -> name.length() == 3)
                        .limit(3);
    
                Stream<String> stream1 = two.stream();
                Stream<String> skip = stream1.filter((name) -> name.startsWith("张"))
                        .skip(2);
    
                Stream<String> concat = Stream.concat(limit, skip);
                ArrayList<Person> p = new ArrayList<>();
                concat.forEach((s) -> p.add(new Person(s)));
    
                for (Person person : p) {
                    System.out.println(person);
                }
    
    
            }
        }
    

    二、方法引用
    在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑?
    2.1 冗余的Lambda场景
    当我们引用一个方法的时候,方法内有一个函数式接口需要lambda表达式的时候,对象和方法已经存在的条件下,我们无需进行方法调用,直接使用方法引用即可。
    改进:

     public class Demo02PrintRef {
            private static void printString(Printable data) {
                data.print("Hello, World!");
            }
            public static void main(String[] args) {
                printString((s) -> System.out.println(s));
                printString(System.out::println);
            }
        }
    

    2.2 方法引用符—— ::
    语义分析
    例如上例中, System.out 对象中有一个重载的 println(String) 方法恰好就是我们所需要的。那么对于
    printString 方法的函数式接口参数,对比下面两种写法,完全等效:
    Lambda表达式写法: s -> System.out.println(s);
    方法引用写法: System.out::println

    2.3 通过对象名引用成员方法
    一个类有一个成员方法:

    public class MethodRefObject {
    			public void printUpperCase(String str) {
    				System.out.println(str.toUpperCase());
    		}
    }
    

    函数式接口:

    @FunctionalInterface
    	public interface Printable {
    		void print(String str);
    }
    

    那么当需要使用这个 printUpperCase 成员方法来替代 Printable 接口的Lambda的时候,已经具有了
    MethodRefObject 类的对象实例,则可以通过对象名引用成员方法,代码为:

    public class Demo04MethodRef {
        private static void printString(Printable lambda) {
            lambda.print("Hello");
        }
        public static void main(String[] args) {
            MethodRefObject obj = new MethodRefObject();
            printString(obj::printUpperCase);
        }
    }
    

    2.4 通过类名称引用静态方法
    由于在 java.lang.Math 类中已经存在了静态方法 abs ,所以当我们需要通过Lambda来调用该方法时,有两种写法。
    首先是函数式接口:

    @FunctionalInterface
    		public interface Calcable {
    			int calc(int num);
    }
    

    lambda表达式:

    public class Demo05Lambda {
        private static void method(int num, Calcable lambda) {
            System.out.println(lambda.calc(num));
        }
        public static void main(String[] args) {
            method(‐10, n ‐> Math.abs(n));
        }
    }
    

    方法引用:

    public class Demo06MethodRef {
        private static void method(int num, Calcable lambda) {
            System.out.println(lambda.calc(num));
        }
        public static void main(String[] args) {
            method(‐10, Math::abs);
        }
    }
    

    2.5 通过super引用成员方法
    如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。首先是函数式接口:

    @FunctionalInterface
    	public interface Greetable {
    			void greet();
    }
    

    父类:

    public class Human {
    		public void sayHello() {
    				System.out.println("Hello!");
    		}
    }
    

    子类,lambda表达式:

    public class Man extends Human {
        @Override
        public void sayHello() {
            System.out.println("大家好,我是Man!");
        }
        //定义方法method,参数传递Greetable接口
        public void method(Greetable g){
            g.greet();
        }
        public void show(){
    //调用method方法,使用Lambda表达式
            method(()‐>{
    //创建Human对象,调用sayHello方法
                    new Human().sayHello();
    });
    //简化Lambda
            method(()‐>new Human().sayHello());
    //使用super关键字代替父类对象
            method(()‐>super.sayHello());
        }
    }
    

    子类,方法引用:

    public class Man extends Human {
        @Override
        public void sayHello() {
            System.out.println("大家好,我是Man!");
        }
        //定义方法method,参数传递Greetable接口
        public void method(Greetable g){
            g.greet();
        }
        public void show(){
            method(super::sayHello);
        }
    }
    

    2.6 通过this引用成员方法
    函数式接口:

    @FunctionalInterface
    		public interface Richable {
    			void buy();
    }
    

    Husband类,lambda:

    public class Husband {
        private void buyHouse() {
            System.out.println("买套房子");
        }
        private void marry(Richable lambda) {
            lambda.buy();
        }
        public void beHappy() {
            marry(() ‐> this.buyHouse());
        }
    }
    

    方法引用:

    public class Husband {
        private void buyHouse() {
            System.out.println("买套房子");
        }
        private void marry(Richable lambda) {
            lambda.buy();
        }
        public void beHappy() {
            marry(this::buyHouse);
      	  }
        }
    

    2.7 类的构造器引用:
    Person类:

    public class Person {
        private String name;
        public Person(String name) {
            this.name = name;
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
    }
    

    创建类的接口:

    public interface PersonBuilder {
    		Person buildPerson(String name);
    	}
    

    方法引用:

    public class Demo10ConstructorRef {
        public static void printName(String name, PersonBuilder builder) {
            System.out.println(builder.buildPerson(name).getName());
        }
        public static void main(String[] args) {
            printName("赵丽颖", Person::new);
        }
    }
    

    2.8 数组的构造器引用

    public class Demo12ArrayInitRef {
        private static int[] initArray(int length, ArrayBuilder builder) {
            return builder.buildArray(length);
        }
        public static void main(String[] args) {
            int[] array = initArray(10, int[]::new);
        }
    }
    
    cs