java Stream流 、方法引用 的 详细阐述

1.Stream流

    在Java 8中,得益于Lambda所带来的函数式编程,引入了一个全新的Stream概念,用于解决已有集合类库既有的弊端

 1.1 引言

    传统集合的多步遍历代码

    几乎所有的集合都支持直接或者间接的遍历操作。

public class Day082 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        list.add("路飞");
        list.add("索隆");
        list.add("山治");
        list.add("乔巴");
        for (String s : list) {
            System.out.println(s);
        }
    }
}

  循环遍历的弊端

      Java 8的Lambda让我们可以更加专注于做什么(What),而不是怎么做(How),这点此前已经结合内部类进行了对比说明。现在,我们仔细体会一下上例代码,可以发现:

                  (1)for循环的语法就是“怎么做”

                  (2)for循环的循环体才是“做什么”

     为什么使用循环?因为要进行遍历。但循环是遍历的唯一方式吗?遍历是指每一个元素逐一进行处理,而并不是从第一个到最后一个顺次处理的循环。前者是目的,后者是方式。

     试想一下,如果希望对集合中的元素进行筛选过滤:

           (1)试想一下,如果希望对集合中的元素进行筛选过滤:

           (2)然后再根据条件二过滤为子集C。

    在java8之前可能是:

public class Day082 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        list.add("路飞22");
        list.add("索隆3333");
        list.add("山治3333");
        list.add("乔巴333");
        List<String> list1=new ArrayList<>();
        for (String s : list) {
            if(s.length()>4){
                list1.add(s);
            }
        }
        for (String s : list1) {
            System.out.println(s);
        }
    }
}

    Stream的更优写法

public class DemoStream {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        list.add("路飞22");
        list.add("索隆3333");
        list.add("山治3333");
        list.add("乔巴333");
        list.stream()
                .filter(s ->s.length()>4)
                .filter(s -> s.length()>5)
                .forEach(System.out::println);
    }
}

      直接阅读代码的字面意思即可完美展示无关逻辑方式的语义:获取流、过滤长度>4、过滤长度>5、逐一打印。代码中并没有体现使用线性循环或是其他任何算法进行遍历,我们真正要做的事情内容被更好地体现在代码中。

1.2 流式思想概述:

     stream流其实就是一个集合元素的函数模型,它并不是集合,也不是数据结构,其本身并不储存任何元素或者地址。

Stream流是一个来自数据源的元素队列:

       元素是特定类型的对象,形成一个队列。java中的stream并不会存储元素,只是计算元素。

      数据源流的来源。可以是集合、数组等。

与collection操作不同,stream流操作还有两个基础的特性。

       Pipelining:中间的操作都会返回流对象本身。这样就可以多个操作串联成一个管道,就像流式风格,这样做可以对操作进行优化,比如说演示和短路。

       内部迭代:以前对结合的遍历都是通过Itertor或者增强for的方式,这些都是显式的在外部迭代,叫做外部迭代。Stream提供了内部迭代的方式。流可以直接调用遍历方法。

     在调用一个流对象时,包括三个步骤:获取一个数据源---->数据转换---->执行操作获取想要的结果。每次进行转换的原有的Stream对象不变,会返回一个新的Stream对象(可以进行多次转换),这就使操作可以像链一样,形成一个管道。

 1.3 获取流

        java.util.stream <T>是java8新加入的最常用的流接口。

       获取一个流非常简单,有以下几种常用的方式: 

          (1)所有的Collection集合可以通过Stream的默认方法,获取流。

           (2)Stream接口的静态方法of可以获取数组对应的流。

     根据Collection获取流:

  首先,java.util.collection接口中加入了default方法stream来获取流。

public class Demostream {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        Stream<String> stream = list.stream();
        Set<String> set=new HashSet<>();
        Stream<String> stream1 = set.stream();
        Vector<String> vector=new Vector<>();
        Stream<String> stream2 = vector.stream();
    }
}

根据Map获取流:

      java.util.Map接口不是Collection的子接口,map结合中的数据结构为K-V不符合流数据的单一特征,因此map集合获取对应的流对象,要分key、value、或者entry等情况。

public class Demostream {
    public static void main(String[] args) {
        Map<String,String> map=new HashMap<>();
        Stream<String> stream = map.keySet().stream();
        Stream<String> stream1 = map.values().stream();
        Stream<Map.Entry<String, String>> stream2 = map.entrySet().stream();
    }
}

   根据数组获取流:

  如果使用的不是集合或者映射而是数组,因为数组对象,不能有添加默认方法,因此Stream接口中提了静态方法of:

public class Demostream {
    public static void main(String[] args) {
        String[] arr={"路飞","索隆","山治","乔巴"};
        Stream<String> arr1 = Stream.of(arr);
    }
}

    of方法的参数其实是一个可变参数,所以支持数组。

1.4 Stream流的常用方法

    Stream流的操作很丰富,这些操作方法分为两种:

     延迟方法:返回值类型仍然是Stream接口自身类型的方法,支持链式编程。(除了终结方法外,都是延迟方法)

     终结方法:返回值类型不再是Stream接口自身类型的方法,不支持链式编程,总结方法有count和forEach方法。

(1) 逐一处理:forEach

       这个方法与for循环中的for-each不同

        void fotEach(Consumer<? super T> action);

该方法接收一个Consumer接口函数,会将每一个流元素交给该函数处理。

Consumer接口

     java.util.function.Consumer<T>接口是一个消费型接口。

   Consumer接口中包含抽象方法void accept(T t),意为消费一个指定泛型的数据。

基本使用:

public class Demostream {
    public static void main(String[] args) {
        String[] arr={"路飞","索隆","山治","乔巴"};
        Stream<String> arr1 = Stream.of(arr);
        arr1.forEach(s-> System.out.println(s));
    }
}
路飞
索隆
山治
乔巴

 (2) 过滤:filter

  可以通过filter方法将一个流转换成另一个子集流。

       Stream<T> filter(Predicate<? super T> predicate);

该接口接收一个Predicate函数式接口(可以是lambda表达式或者方法引用)参数作为筛选条件。

  Predicate接口: 

        Predicate接口中有一个抽象方法:

             boolean test(T t);

    test方法会产生一个boolean值结果,代表指定的条件是否满足。如果结果为true,那么filter方法将会留用元素,如果结果为false方法会舍弃元素。

基本使用:

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("路飞", "路西", "索隆");
        Stream<String> stream1 = stream.filter(s -> s.startsWith("路"));
        stream1.forEach(System.out::println);
    }
}
路飞
路西

 (3) 映射:map

   如果将流中的元素映射到另一个流中,可以使用map的方法。

      <R> Stream<R> map(Function<? super T, ? extends R> mapper);

该接口需要一个Function函数式接口,可以将当前流中的T类型数据转换成另一种R类型的流。

   Function函数式接口

     java.util.stream.Function 函数式接口,其中唯一的抽象方法为:

           R apply (T t);

这可以将一种T类型转换成R类型,这种转换的动作,就称为“映射”。

基本使用:

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("10", "20", "10");
        Stream<Integer> stream1 = stream.map(s ->Integer.parseInt(s));
        stream1.forEach(System.out::println);
    }
}
10
20
10

(4)统计个数

   正如旧集合Collection 当中的size 方法一样,流提供count 方法来数一数其中的元素个数:

           long count();

   该方法返回一个long值代表元素个数(不再像旧集合那样是int值)。基本使用:

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("张三", "张六", "李二");
        Stream<String> stream1= stream.filter(s -> s.startsWith("张"));
        System.out.println(stream1.count());
    }
}
2

     (5)取用前几个:limit

   limit方法可以对流进行截取,只取用前几个。

      Stream<T> limit (long maxSize);

参数是一个long型,如果集合当前长度大于参数则进行截取;否则不进行操作。基本使用

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("张三", "张六", "李二");
        Stream<String> stream1 = stream.limit(2);
        stream1.forEach(System.out::println);
    }
}
张三
张六

    (6)跳过前几个:skip

如果希望跳过前几个元素,可以使用skip 方法获取一个截取之后的新流

     Stream <T> skip(long n);

如果流的当前长度大于n,则跳过前n个;否则将会得到一个长度为0的空流。基本使用:

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("张三", "张六", "李二");
        Stream<String> stream1 = stream.skip(1);
        stream1.forEach(System.out::println);
    }
}
张六
李二

  (7)组合:concat

如果有两个流,希望合并成为一个流,那么可以使用Stream 接口的静态方法concat :

   static <T> Stream<T> concat(Stream<? extends T> a, Stream<? extends T> b)

public class Demostream {
    public static void main(String[] args) {
        Stream<String> stream = Stream.of("张三", "张六", "李二");
        Stream<String> stream1 = Stream.of("张二", "张四", "李以");
        Stream<String> stream3 = Stream.concat(stream, stream1);
        stream3.forEach(System.out::println);
    }
}
张三
张六
李二
张二
张四
李以

  1.5 集合元素处理练习

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

                   1. 第一个队伍只要名字为3个字的成员姓名;存储到一个新集合中。

                   2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。

                  .3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。

                   4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。

                   5. 将两个队伍合并为一个队伍;存储到一个新集合中。

                   6. 根据姓名创建Person 对象;存储到一个新集合中。

                   7. 打印整个队伍的Person对象信息。

     使用传统的模式:

         1. 构建Person类

public class Person {
    private String name;

    public Person() {
    }

    public Person(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }

    public void setName(String name) {
        this.name = name;
    }
}

     2,实现代码

public class DemoArraylist {
    public static void main(String[] args) {
        //第一支队伍
        ArrayList<String> list=new ArrayList<>();
        list.add("迪丽热巴");
        list.add("宋远桥");
        list.add("苏星河");
        list.add("石破天");
        list.add("石中玉");
        list.add("老子");
        list.add("庄子");
        list.add("洪七公");
        //第二只队伍
        ArrayList<String> list1=new ArrayList<>();
        list1.add("古力娜扎");
        list1.add("张无忌");
        list1.add("赵丽颖");
        list1.add("张三丰");
        list1.add("尼古拉斯赵四");
        list1.add("张天爱");
        list1.add("张二狗");
        // 第一个队伍只要名字为3个字的成员姓名;
        ArrayList<String> listA=new ArrayList<>();
        for (String s : list) {
            if(s.length()==3){
                listA.add(s);
            }
        }
        // 第一个队伍筛选之后只要前3个人;
        ArrayList<String> listB=new ArrayList<>();
        for (int i = 0; i < 3; i++) {
            listB.add(listA.get(i));
        }

        // 第二个队伍只要姓张的成员姓名;
        ArrayList<String> list1A=new ArrayList<>();
        for (String s : list1) {
            if(s.startsWith("张")){
                list1A.add(s);
            }
        }
        // 第二个队伍筛选之后不要前2个人;
        ArrayList<String> list1B=new ArrayList<>();
        for (int i = 2; i < list1B.size(); i++) {
            list1B.add(list1A.get(i));
        }
        // 将两个队伍合并为一个队伍;
        ArrayList<String> newlist=new ArrayList<>();
        newlist.addAll(listB);
        newlist.addAll(list1B);
        // 根据姓名创建Person对象;
        ArrayList<Person> newlist1=new ArrayList<>();
        for (String s : newlist) {
            newlist1.add(new Person(s));
        }
        // 打印整个队伍的Person对象信息。
        for (Person person : newlist1) {
            System.out.println(person);
        }
    }
}
Person{name='宋远桥'}
Person{name='苏星河'}
Person{name='石破天'}
Person{name='张天爱'}
Person{name='张二狗'}

   使用Stream方式处理:

public class DemoStream1 {
    public static void main(String[] args) {
        //第一支队伍
        ArrayList<String> list = new ArrayList<>();
        list.add("迪丽热巴");
        list.add("宋远桥");
        list.add("苏星河");
        list.add("石破天");
        list.add("石中玉");
        list.add("老子");
        list.add("庄子");
        list.add("洪七公");
        //第二只队伍
        ArrayList<String> list1 = new ArrayList<>();
        list1.add("古力娜扎");
        list1.add("张无忌");
        list1.add("赵丽颖");
        list1.add("张三丰");
        list1.add("尼古拉斯赵四");
        list1.add("张天爱");
        list1.add("张二狗");
        //第一个队伍只要名字为3个字的成员姓名;第一个队伍筛选之后只要前3个人;
        Stream<String> listA = list.stream().filter(s -> s.length() == 3).limit(3);
        // 第二个队伍只要姓张的成员姓名;第二个队伍筛选之后不要前2个人;
        Stream<String> list1A = list1.stream().filter(s -> s.startsWith("张")).skip(2);
        //将两个队伍合并为一个队伍;根据姓名创建Person对象;打印整个队伍的Person对象信息。
        Stream.concat(listA, list1A).map(Person::new).forEach(System.out::println);
    }
}
Person{name='宋远桥'}
Person{name='苏星河'}
Person{name='石破天'}
Person{name='张天爱'}
Person{name='张二狗'}

 上述的map中的lambda表达式使用的简便操作,也可这样写 map(s -> new Person(s))。

2.方法引用

    在使用lambda表达式中,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在lambda中所指定的操作方案,已经有地方存在相同的方案,我们是够还有必要进行重写?

  2.1 冗余的Lambda场景

   我们来看一个简单的函数式接口以应用lambda表达式:

@FunctionalInterface
public interface println {
    void print(String s);
}

      在println接口中唯一的抽象方法print方法接收一个参数,目的就是为了打印这个参数,用lambda表达式很简单。

public class DemoPrintln {
    public static void println(println p){
        p.print("函数式接口");
    }

    public static void main(String[] args) {
        println(s -> System.out.println(s));
    }
}
函数式接口

     其实println方法只管调用pritlnj接口中的print方法,并不会关心print方法的具体显示逻辑会将字符串打印到哪里。在main方法中通过lambda表达式指定了函数接口print的具体实现步骤:拿到String数据后,在控制台输出。

 2.2 问题分析

     这段代码的问题在于,对字符串进行控制台打印输出的操作方案,明明有了现有的实现,那就是System.out对象中的println(String)方法。既然lambda表达式希望调用的事println(String)方法,那何必自己在重写呢?

  2.3 用方法引用改进代码

public class DemoPrintln {
    public static void println(println p){
        p.print("函数式接口");
    }

    public static void main(String[] args) {
        println(System.out::println);
    }
}
函数式接口

    双冒号 “::”写法,称为方法引用,这是一种新的语法。

2.4 方法引用符

      双冒号“::”为引用运算符,它所在的表达式称为方法引用。如果lambda要表达的函数方案已经存在与某在方法的实现中,那么就可以用双冒号来引用作为方法的替代者。

语法分析:

          例如上例中, System.out 对象中有一个重载的println(String) 方法恰好就是我们所需要的。那么对于printString 方法的函数式接口参数,对比下面两种写法,完全等效;

      (1)Lambda表达式写法: s -> System.out.println(s);

       (2)方法引用写法: System.out::println

  第一种语义是指:拿到参数之后经Lambda之手,继而传递给System.out.println 方法去处理。

 第二种等效写法的语义是指:直接让System.out 中的println 方法来取代Lambda。两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。

注::Lambda 中 传递的参数 一定是方法引用中 的那个方法可以接收的类型,否则会抛出异常。

推导与省略:

     如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式——它们都将被自动推导。而如果使用方法引用,也是同样可以根据上下文进行推导。

      函数式接口是Lambda的基础,而方法引用是Lambda的孪生兄弟。

下面这段代码将会调用println 方法的不同重载形式,将函数式接口改为int类型的参数:

@FunctionalInterface
public interface println {
    void print(int s);
}

由于上下文变了之后可以自动推导出唯一对应的匹配重载,所以方法引用没有任何变化:

public class DemoPrintln {
    public static void println(println p){
        p.print(10);
    }

    public static void main(String[] args) {
        println(System.out::println);
    }
}
10

     这次方法引用将会自动匹配到println(int) 的重载形式。

2.5 通过对象名引用成员方法

    如果在一个类中定义了一种输出的方法;

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

    此时有一个函数式接口:

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

   若要使用Methodobject类中的方法,来替代printable 接口中的lambda是,就需要Methodobject类的实现类对象,通过对象名引用成员方法。

public class DemoMethodobj {
    public static void println(Printable p){
        p.print("hello");
    }

    public static void main(String[] args) {
        Methodobject mobj=new Methodobject();
        println(mobj::printUpperCase);
    }
}
HELLO

 2.6 通过类名称引用静态方法。

     由于在java.lang.Math 类中已经存在了静态方法abs ,所以当我们需要通过Lambda来调用该方法时,有两种写法。

      首先定义 函数式接口:

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

第一种写法用lambda表达式:

public class DemoLambda {
    private static void method(int num,Calcable c){
        System.out.println(c.calc(num));
    }

    public static void main(String[] args) {
        method(-10,num->Math.abs(num));
    }
}
10

   第二种使用方法引用:

public class DemoLambda {
    private static void method(int num,Calcable c){
        System.out.println(c.calc(num));
    }

    public static void main(String[] args) {
        method(-10,Math::abs);
    }
}

 其中:Lambda表达式: num-> Math.abs(num)  与 方法引用: Math::abs  是等价的。

2.7 通过super引用成员变量

   如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。

 首先是函数式接口:

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

      定义父类Human:

public class Human {
    public void sayhello(){
        System.out.println("hello");
    }
}

     定义子类Man,其中使用了lambda表达式

public class Man extends Human {
    @Override
    public void sayhello(){
        System.out.println("我是子类的hello");
    }
    //定义方法,参数传递接口
    public void method(Greetable greetable){
        greetable.greet();
    }
    public void show(){
        //调用method方法,使用lambda表达式
        method(()->{
            //创建Human对象,条用sayhello方法
            new Human().sayhello();
        });
        //简化lambda
        method(()->new Human().sayhello());
        //使用super关键字代替父类对象
        method(()->super.sayhello());
        //使用this关键字代替父类对象
        method(()->this.sayhello());
    }

    public static void main(String[] args) {
        Man m=new Man();
        m.show();
    }
}
hello
hello
hello
我是子类的hello

  定义子类Man,其中使用了方法引用

public class Man1 extends Human {
    @Override
    public void sayhello(){
        System.out.println("我是子类的hello");
    }
    //定义方法,参数传递接口
    public void method(Greetable greetable){
        greetable.greet();
    }
    public void show(){
        //使用super关键字代替父类对象
        method(super::sayhello);
        //使用this关键字代替父类对象
        method(()->this.sayhello());
    }

    public static void main(String[] args) {
        Man1 m=new Man1();
        m.show();
    }
}
hello
我是子类的hello

      2.8 通过this引用成员方法

public class Man1 extends Human {
    @Override
    public void sayhello(){
        System.out.println("我是子类的hello");
    }
    //定义方法,参数传递接口
    public void method(Greetable greetable){
        greetable.greet();
    }
    public void show(){
        //使用this关键字代替父类对象
        method(()->this.sayhello());
        //通过方法引用
        method(this::sayhello);
    }

    public static void main(String[] args) {
        Man1 m=new Man1();
        m.show();
    }
}
我是子类的hello
我是子类的hello

  2.9 类的构造器引用

     由于构造器的名称与类名完全一样,并不固定。所以构造器引用使用类名称::new 的格式表示。

首先是一个简单的Person 类:

public class Person {
    private String name;

    public Person() {
    }

    public Person(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }

    public void setName(String name) {
        this.name = name;
    }
}

    用来创建Person对象的函数式接口:

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

  通过lambda表达式,使用和这个函数式接口:

public class DemoPerson {
    public static void printName(String name,PersonBuilder pb){
        System.out.println(pb.buildPerson(name).getName());
    }

    public static void main(String[] args) {
        printName("路飞",(name)->new Person(name));
    }
}
路飞

  但是通过构造器引用,有更好的写法:

public class DemoPerson {
    public static void printName(String name,PersonBuilder pb){
        System.out.println(pb.buildPerson(name).getName());
    }

    public static void main(String[] args) {
        printName("路飞",Person::new);
    }
}
路飞

    Lambda表达式: name -> new Person(name) 和方法引用: Person::new  等效。

2.10 数组的构造器引用

    数组也是Object 的子类对象,所以同样具有构造器,只是语法稍有不同。如果对应到Lambda的使用场景中时,

     需要一个函数式接口:

public interface Arraybuilder {
    int[] buildArray();
}

      在应用该接口的时候,可以通过Lambda表达式:

public class Demo {
    private static int[] inttArray(int length,Arraybuilder abr){
        return abr.buildArray(length);
    }

    public static void main(String[] args) {
        inttArray(10,length -> new int[length]);
    }
}

   但是更好的写法是使用数组的构造器引用:  

public class Demo {
    private static int[] inttArray(int length,Arraybuilder abr){
        return abr.buildArray(length);
    }

    public static void main(String[] args) {
        inttArray(10,int[]::new);
    }
}

   Lambda表达式: length -> new int[length]  与方法引用: int[]::new  等效。

 

   

 

 

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值