Java基础回顾-JDK8新特性【函数式接口、Stream流、方法引用】

目录

JDK8新特性

函数式接口

概念

函数式接口在Java中是指:有且仅有一个抽象方法的接口

函数式接口,即适用于函数式编程场景的接口。而Java中的函数式编程体现就是Lambda,所以函数式接口就是可以适用于Lambda使用的接口。只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利地进行推导。

备注:“语法糖"是指使用更加方便,但是原理不变的代码语法。例如在遍历集合时使用的for-each语法,其实底层的实现原理仍然是迭代器,这便是"语法糖"。从应用层面来讲,Java中的Lambda可以被当做是匿名内部类的“语法糖”,但是二者在原理上是不同的。

格式

只要确保接口中有且仅有一个抽象方法即可:

修饰符 interface 接口名称{
	public abstract 返回值类型 方法名称(可选参数信息);
    //其他非抽象方法内容
}

由于接口当中抽象方法的public abstract 是可以省略的,所以定义一个函数式接口很简单

public interface MyFunctionalInterface {
	void myMethod( );
}

函数式接口的使用

package MyDiyInterface;
/*
    函数式接口的使用:一般可以作为方法的参数和返回值类型
 */
public class DemosFunctionalInterfaceTest {
    //定义一个方法,参数使用函娄数式接口MyFunctionalInterface
    public static void show(MyFunctionalInterface myInter) {
        myInter.method();
    }

    public static void main(String[] args) {
        //调用show方法,方法的参数是一个接口,所以可以传递接口的实现类对象
        show(new MyFunctionalInterfaceImpl());

        //调用show方法,方法的参数是一个接口,所以可以传递接口的匿名内部类
        show(new MyFunctionalInterface() {
            @Override
            public void method() {
                System.out.println("使用匿名内部类重写接口中的抽象方法");
            }
        });

        //调用show方法,方法的参数是一个函数式接口,所以我们可以春娣Lambda表达式
        show(()->{
            System.out.println("使用Lambda表达式重写接口中的抽象方法");
        });
        //简化Lambda表达式代码
        show(()-> System.out.println("使用Lambda表达式重写接口中的抽象方法"));
    }

}


Lambda的延迟执行

有些场景的代码执行后,结果不一定会被使用,从而造成性能浪费。而fambda表达式是延迟执行的,这正好可以作为解决方案,提升性能。

性能浪费的日志案例

注:日志可以帮助我们快速的定位问题,记录程序运行过程中的情况,以便项目的监控和优化。
一种典型的场景就是对参数进行有条件使用,例如对日志消息进行拼接后,在满足条件的情况下进行打印输出︰

package MyDiyInterface;
/*
    日志案例
    发现以下代码存在的一些性能浪费的问题
    
    调用showLog方法,传递的第二个参数是一个拼接后的字符串
    
    先把字符串拼接好,然后在调用showLog方法
    
    showLog方法中如果传递的日志等级不是1级
    那么就不会是如此拼接后的字符串
    所以感觉字符串就白拼接了,存在了浪费
 */
public class Demo01Logger {
    private static void showLog(int level,String msg){
        if ( level == 1) {
            System.out.println(msg);
        }
    }
    public static void main( String[] args){
        String msgA = "Hello";
        String msgB = "world";
        String msgC = "Java";
        showLog(1,msgA + msgB + msgC);
    }
}

日志案例代码优化【使用Lambda表达式】

package MyDiyInterface;
/*
    使用Lambda优化日志案例
    Lambda的特点:延迟加载
    Lambda的使用前提,必须存在函数式接口
 */
public class Demo02Lambda {
    //定义一个显示日志的方法,方法的参数传递日志的等级和MessageBuilder接口
    public static void showLog(int level,MessageBuilder mb){
        //对日志的等级进行判断,如果是1级,则调用MessageBuilder接口中的builderMessage方法
        if (level == 1){
            System.out.println(mb.builderMessage());
        }
    }

    public static void main(String[] args) {
        //定义三个日志信息
        String msgA = "Hello ";
        String msgB = "world ";
        String msgC = "Java ";
        //调用showLog方法,参数MessageBuilder是一个函数式接口,所以可以传递Lambda表达式
        showLog(1,()->{
            //返回一个拼接好的字符串
            return msgA+msgB+msgC;
        });

        /*
            使用Lambda表达式作为参数传递,仅仅是把参数传递到showLog方法中
            只有满足条件,日志的等级是1级
                才会调用接口NessageBuilder中的方法buiLderMessage
                才会进行字符串的拼接
            如果条件不满足,日志的等级不是1级
                那么NessageBuilder接口中的方法builderMessage也不会执行
                所以拼接字符串的代码也不会执行
                所以不会存在性能的浪费
         */

    }

}


使用Lambda作为参数个返回值

如果抛开实现原理不说,Java中的Lambda表达式可以被当作是匿名内部类的替代品。如果方法的参数是一个函数式接口类型,那么就可以使用Lambda表达式进行替代使用Lambda表达式作为方法参数,其实就是使用函数式接口作为方法参数。
例如java.lang .Runnable接口就是一个函数式接口假设有一个startThread方法使用该接口作为参数,那么就可以使用Lambda进行传参。这种情况其实和Thread类的构造方法参数为Runnable没有本质区别。

public class DemosRunnable {
    //定义一个方法,startThread,方法的参数使用函数式接口Runnable
    public static void startThread(Runnable run){
        //开启多线程
        new Thread(run).start();
    }

    public static void main(String[] args) {
        //调用startThread方法,方法的参数是一个接口,那么我们就可以传递这个接口的匿名内部类
        startThread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName()+"--->"+"线程启动了");
            }
        });

        //用Lambda表达式优化上面的代码
        //调用startThread方法,方法的参数是一个函数式接口,那么我们就可以传递Lambda表达式
        startThread(()->System.out.println(Thread.currentThread().getName()+"--->"+"线程启动了")    );
    }
}

类似地,如果一个方法的返回值类型是一个函数式接口,那么就可以直接返回一个Lambda表达式。当需要通过一个方法来获取一个java.util.Comparator 接口类型的对象作为排序器时,就可以调该方法获取。

import java.util.Arrays;
import java.util.Comparator;

/*
    类似地,如果一个方法的返回值类型是一个函数式接口,那么就可以直接返回一个Lambda表达式。
    当需要通过一个方法来获取一个java.util.Comparator 接口类型的对象作为排序器时,就可以调该方法获取。
 */
public class DemosRunnable {
    //定义一个方法,方法的返回值类型使用函数式接口Comparator
    public static Comparator<String> getComparator(){
        //方法的返回值类型是一个接口,那么我们可以返回这个接口的匿名内部类
       /*return new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                //按照字符串的降序排序
                return o2.length()-o1.length();
            }
        };*/
        //方法的返回值类型是一个函数式接口,那么我们可以返回一个Lambda表达式
        return (o1, o2)->o2.length()-o1.length();

    }

    public static void main(String[] args) {
        //创建一个字符串数组
        String[] arr = {"a","b","CCC","dd","eeeeeee","ffff"};
        //输出排序前的数组
        System.out.println(Arrays.toString(arr));//[a, b, CCC, dd, eeeeeee, ffff]
        //调用Arrays中的sort方法,对字符串数组进行排序
        Arrays.sort(arr,getComparator());
        //输出排序后的数组
        System.out.println(Arrays.toString(arr));//[eeeeeee, ffff, CCC, dd, a, b]
    }
}

常用函数式接口

Supplier接口

java.util.function. supplier接口仅包含一个无参的方法:T get()。用来获取一个泛型参数指定类型的对象数据。由于这是一个函数式接口,这也就意味着对应的Lambda表达式需要对外提供"一个符合泛型类型的对象数据。

Supplier接口被称之为生产型接口,指定接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据

package MyDiyInterface;
import java.util.function.Supplier;

/*
    常用的函威数式接口
        java.util. function .SuppLier<T>接口仅包含一个无参的方法: T get()。用来获取一个泛型参数指定类型的对象数据。
        Supplier<T>接口被称之为生产型接口,指定接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据
 */
public class DemosSupplier {
    //定义一个方法,方法的参数传递SuppLier<T>接口,泛型指定string类型,get方法就会返回一个String字符串
    public static String getString(Supplier<String> sup){
        return sup.get();//返回一个字符串

    }

    public static void main(String[] args) {
        //调用getString方法,方法得参数Supplier是一个函数值接口,所以可以传递Lambda表达式
        String str = getString(() ->"字符串jasjdkahsjd");//生产一个字符串并返回

        System.out.println(str);
    }
}

练习题:求数组中元素的最大值

使用Supplier 接口作为方法参数类型,通过Lambda表达式求出int数组中的最大值。提示:接口的泛型请使用java.lang.Integer类。

import java.util.function.Supplier;

public class DemosMax {
    public static Integer getMax(Supplier<Integer> sup){
        return sup.get();
    }

    public static void main(String[] args) {
        Integer arr[] = {5,423,453,34,5,234,52};

        Integer max1 = getMax(() -> {
            int max = arr[0];
            /*for (int i = 0; i < arr.length-1; i++) {
                if (max > arr[i + 1]) {
                    max = max;
                } else {
                    max = arr[i + 1];
                }
            }*/
            for (Integer integer : arr) {
                if (integer>max){
                    max = integer;
                }
            }
            return max;
        });
        System.out.println("数组中最大值为:"+max1);
    }
}


Consumer接口【消费数据接口】

java.util.function.Consumer 接口则正好与Supplier接口相反,它不是生产一个数据,而是消费一个数据,其数据类型由泛型决定。

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

import java.util.function.Consumer;
import java.util.function.Supplier;

public class DemosConsumer {
    /*
        定义一个方法
        方法的参数传递一个字符串的姓名
        方法的参数传递consumer接口,泛型使用string可以使用Consumer接口消费字符串的姓名
     */
    public static void method(String name,Consumer<String> cons){
        cons.accept(name);
    }

    public static void main(String[] args) {
        //调用method方法,传递字符串姓名,方法的另一个参数是Consumer接口,是一个函数式接口,所以可以传递Lambda表达式
        method("牛牛niu",(String name)->{
            //对传递的字符串进行消费
            //消费方式:直接输出字符串
            System.out.println(name);
            //消费方式:把字符串进行反转输出
            String reName = new StringBuilder(name).reverse().toString();
            System.out.println(reName);
        });
    }
}

Consumer接口中的默认方法:addThen

如果一个方法的参数和返回值全都是Consumer类型,那么就可以实现效果∶消费数据的时候,首先做一个操作,然后再做一个操作,实现组合。而这个方法就是Consumer 接口中的default方法 and Then 。下面是JDK的源代码∶

default consumer<T> andThen(Consumer<? super T> after){
    objects.requireNonNull(after);
    return (T t) -> { accept(t); after.accept(t);
}

备注: java.util.Objects 的 requireNonNull静态方法将会在参数为null时主动抛出 NullPointerException异常。这省去了重复编写if语句和抛出空指针异常的麻烦。

要想实现组合,需要两个或多个Lambda表达式即可,而 andThen 的语义正是"一步接一步"操作。例如两个步骤组合的情况:

两个lambda的情况:

import java.util.Locale;
import java.util.function.Consumer;

/*
    consumer接口的黑X认方法andThen
    作用:需要两个consumer接口,可以把两个Consumer接口组合到一起,在对数据进行消费例如:

    consumer<String> con1
    consumer<string> con2
    String s = "heLLo"";
    coni.accept(s);
    con2.accept(s);
    连接两个Consuwer接口,再进行消费
    coni.andThen( con2).accept(s);谁写前边谁先消费
 */
public class DemosAndThen {
    //定义一个方法,方法的参数传递一个字符串和两个Consumer接口, Consumer接口的泛型使用字符串
    public static void method(String str, Consumer<String> cons1, Consumer<String> cons2){
//        cons1.accept(str);
//        cons2.accept(str);
        //使用addThen方法,把两个Consumer接口连接在一起,然后再进行消费数据
        cons1.andThen(cons2).accept(str);///con1连接con2,先执行con1消费数据,再执行con2消费数据
    }

    public static void main(String[] args) {
        //调用method方法,传递一个字符串、两个Lambda表达式
        method("字符串ABCdjadka",(String str)->{
            //消费str,消费方式:把字符串变为大写
            System.out.println(str.toUpperCase());
        },(String str)->{
            //消费str,消费方式:把字符串变为小写
            System.out.println(str.toLowerCase());
        });

        //代码优化
        method("字符串ABCdjadka", (str)->System.out.println(str.toUpperCase()) , (str)->System.out.println(str.toLowerCase()) );
    }
}


两个以上lambda的情况:

import java.util.Locale;
import java.util.function.Consumer;

/*
    consumer接口的黑X认方法andThen
    作用:需要两个consumer接口,可以把两个Consumer接口组合到一起,在对数据进行消费例如:

    consumer<String> con1
    consumer<string> con2
    String s = "heLLo"";
    coni.accept(s);
    con2.accept(s);
    连接两个Consuwer接口,再进行消费
    coni.andThen( con2).accept(s);谁写前边谁先消费
 */
public class DemosAndThen {
    //定义一个方法,方法的参数传递一个字符串和两个Consumer接口, Consumer接口的泛型使用字符串
    public static void method(String str, Consumer<String> cons1, Consumer<String> cons2, Consumer<String> cons3){
//        cons1.accept(str);
//        cons2.accept(str);
        //使用addThen方法,把两个Consumer接口连接在一起,然后再进行消费数据
        cons1.andThen(cons2).andThen(cons3).accept(str);///con1连接con2,先执行con1消费数据,再执行con2消费数据,再执行con3消费数据
    }

    public static void main(String[] args) {
        //调用method方法,传递一个字符串、两个Lambda表达式
        method("字符串ABCdjadka",(String str)->{
            //消费str,消费方式:把字符串变为大写
            System.out.println(str.toUpperCase());
        },(String str)->{
            //消费str,消费方式:把字符串变为小写
            System.out.println(str.toLowerCase());
        },(String str)->{
            //消费str,消费方式:直接输出
            System.out.println(str);
        });
        System.out.println("==============================");
        //代码优化
        method("字符串ABCdjadka",
                (str)->System.out.println(str.toUpperCase()),
                (str)->System.out.println(str.toLowerCase()),
                (str)->System.out.println(str) );
    }
}


练习题:格式化打印信息

下面的字符串数组当中存有多条信息,请按照格式"姓名:XX。性别:XX。"的格式将信息打印出来。要求将打印姓名的动作作为第一个Consumer 接口的Lambda实例,将打印性别的动作作为第二个Consumer 接口的Lambda实例,将两个Consumer接口按照顺序"拼接”到一起。

import java.util.function.Consumer;

/*
    下面的字符串数组当中存有多条信息,请按照格式"姓名∶XX。性别:xxX."的格式将信息打印出来。

    要求
    将打印姓名的动作作为第一个consumer 接口的Lambda实例,
    将打印性别的动作作为第二个consumer 接口的Lambda实例,
    将两个Consumer接口按照顺序"拼接”到一起。
 */
public class DemosConsumer2 {

    public static void method(String[] arr, Consumer<String> cons1, Consumer<String> cons2){
        for (String s : arr) {
            cons1.andThen(cons2).accept(s);
        }

    }
    public static void main(String[] args) {
        String[] arr = {"阿牛,男","阿瓜,女","阿巴,未知"};
        method(arr,(s)->{
            System.out.print("姓名:"+s.split(",")[0]+"。");
        },(s)->{
            System.out.println("性别:"+s.split(",")[1]+"。");
        });
        System.out.println("==================================");
        //代码优化
        method(arr,
                (s)-> System.out.print("姓名:"+s.split(",")[0]+"。"),
                (s)-> System.out.println("性别:"+s.split(",")[1]+"。"));


    }



}


Predicate接口【判断接口】

有时候我们需要对某种类型的数据进行判断,从而得到一个boolean值结果。这时可以使用java.util.function.Predicate接口。

抽象方法: test

Predicate接口中包含一个抽象方法:boolean test(T t)。用于条件判断的场景∶

import java.util.function.Predicate;

/*
    java.util.function . Predicate<T>接口
        作用:对某种数据类型的数据进行判断,结果返回一个booLean值
        Predicate接口中包含一个抽象方法:
            boolean test(T t):用来对指定数据类型数据进行判断的方法
                结果:
                    符合条件,返回true
                    不符合条件,返回false
 */
public class DemosPredicate {
    /*
        定义一个方法
            参数传递一个string类型的字符串
            传递一个Predicate接口,泛型使用string
            使用Predicate中的方法test对字符串进行判断,并把判断的结果返回
     */
    public static boolean method(String str, Predicate<String> pre){
        return pre.test(str);
    }

    public static void main(String[] args) {
        /*boolean b = method("字符串jdhfahsfkasjdlhawidhlaw", (String s) -> {
            //对参数传递的字符串进行判断,判断字符串的长度是否大于5,并把判断的结果返回
            return s.length() > 5;
        });
        System.out.println(b);*/
        //代码优化
        boolean b = method("字符串jdhfahsfkasjdlhawidhlaw",(s)->s.length() > 5);
        System.out.println(b);
    }
}
Predicate接口中的3个默认方法
默认方法:and

既然是条件判断,就会存在与、或、非三种常见的逻辑关系。其中将两个Predicate条件使用“与"逻辑连接起来实现“并且"的效果时,可以使用default方法and。其JDK源码为∶

default Predicate<T> and( Predicate<? super T> other) {
	0bjects.requireNonNull(other);
	return (t) -> test(t) && other.test(t);
}

判断一个字符串既要包含大写字母H,又要包含大写字母W,且长度大于5

import java.util.function.Predicate;

/*
    逻辑表达式:可以连接多个判断的条件
    &&:与运算符,有false则false
    ||:或运算符,有true则true
    !:非(取反)运算符,非真则假,非假则真

    需求:判断一个字符串,有两个判断的条件
        1.判断字符串的长度是否大于5
        2.判断字符串中是否包含a
    两个条件必须同时满足,我们就可以使用&&运算符连接两个条件

    Predicate接口中有一个方法and ,表示并且关系,也可以用于连接两个判断条件
    default Predicate<T> and(Predicate<? super T> other) {
        objects.requireNonNulL ( other);
        return (t) -> this.test(t) && other.test(t);
    }
    方法内部的两个判断条件,也是使用&&运算符连接起来的

 */
public class DemosPredicateAdd {
    /*
        定义一个方法,方法的参数,传递一个字符串
        传递两个Predicate接口
            一个用于判断字符串的长度是否大于5
            一个用于判断字符串中是否包含a
            两个条件必须同时满足
     */
    public static boolean method(String str, Predicate<String> pre1,Predicate<String> pre2,Predicate<String> pre3){
        return pre1.test(str)&&pre2.test(str)&&pre3.test(str);
    }

    public static boolean method2(String str, Predicate<String> pre1,Predicate<String> pre2,Predicate<String> pre3){
        return pre1.and(pre2).and(pre3).test(str);//用and方法也可以
    }

    public static void main(String[] args) {
//        String s1 = "ajHshkWdu";
        String s2 = "ajhshkWdu";
//        String s3 = "ajasHkWdu";
        boolean b1 = method("ajhshkWdu", (String str) -> {
            return str.length()>5;
        }, (String str) -> {
            return str.contains("H");
        }, (String str) -> {
            return str.contains("W");
        });
        System.out.println(b1);//false

        //优化Lambda表达式
        boolean b2 = method(s2, (str) ->str.length()>5,(str) ->str.contains("H"),(str) ->str.contains("W"));
        System.out.println(b2);//false

        boolean b3 = method2("HkWsadsad", (str) ->str.length()>5,(str) ->str.contains("H"),(str) ->str.contains("W"));//用and方法也可以
        System.out.println(b3);//true
    }
}


默认方法:or
import java.util.function.Predicate;

/*
    需求:判断一个字符串,有两个手断的条件
        1 .判断字符串的长度是否大于5
        2.岁断字符串中是否包含a
    满足一个条件即可,我们就可以使用//运算符连接两个条件
    Predicate接口中有一个方法or,表示或者关系,也可以用于连接两个判断条件
    default Predicate<T> or ( Predicate<? super T> other){
        objects.requireNonNulL(other);
        return (t) -> test (t) || other.test(t);
    }
    方法内部的两个判断条件,也是使用||运算符连接起来的

 */
public class DemosPredicateOr {
    /*
        定义一个方法,方法的参数,传递一个字符串
        传递两个Predicate接口
            一个用于判断字符串的长度是否大于5
            一个用于判断字符串中是否包含a
            满足一个条件即可
     */
    public static boolean method(String str, Predicate<String> pre1,Predicate<String> pre2,Predicate<String> pre3){
        return pre1.test(str)||pre2.test(str)||pre3.test(str);
    }

    public static boolean method2(String str, Predicate<String> pre1,Predicate<String> pre2,Predicate<String> pre3){
        return pre1.or(pre2).or(pre3).test(str);//用and方法也可以
    }

    public static void main(String[] args) {
//        String s1 = "ajHshkWdu";
        String s2 = "ajhshkWdu";
//        String s3 = "ajasHkWdu";
        boolean b1 = method("aaa", (String str) -> {
            return str.length()>5;
        }, (String str) -> {
            return str.contains("H");
        }, (String str) -> {
            return str.contains("W");
        });
        System.out.println(b1);

        //优化Lambda表达式
        boolean b2 = method(s2, (str) ->str.length()>5,(str) ->str.contains("H"),(str) ->str.contains("W"));
        System.out.println(b2);

        boolean b3 = method2("HkWsadsad", (str) ->str.length()>5,(str) ->str.contains("H"),(str) ->str.contains("W"));//用and方法也可以
        System.out.println(b3);
    }
}


默认方法:negate
import java.util.function.Predicate;

/*
    需求:判断一个字符串长度是否大于5
        如果字符串的长良大于5,那返回false
        如果字特串的长度不大于5,那么返回true
    所以我们可以使用取反符号/对判断的结果进行取反

    Predicate接口中有一个方法negate,也表示职反的意思
    default Predicate<T negate(){
        return (t) ->!test(t);
    }
 */
public class DemosPredicateNegate {
    /*
        定义一个方法,方法的参数,传递一个字符串
        传递一个Predicate接口
            判断字符串的长度是否大于5,对判断结果取反
     */
    public static boolean method(String str, Predicate<String> pre1){
        return !pre1.test(str);
    }

    public static boolean method2(String str, Predicate<String> pre1){
        return pre1.negate().test(str);//用and方法也可以
    }

    public static void main(String[] args) {
//        String s1 = "ajHshkWdu";
        String s2 = "ajhshkWdu";
//        String s3 = "ajasHkWdu";
        boolean b1 = method("aaa", (String str) -> {
            return str.length()>5;
        });
        System.out.println(b1);//true

        //优化Lambda表达式
        boolean b2 = method(s2, (str) ->str.length()>5);
        System.out.println(b2);//false

        boolean b3 = method2("HkWsadsad", (str) ->str.length()>5);//用and方法也可以
        System.out.println(b3);//false
    }
}


练习题:集合信息删选
//Lambda表达式简化
        ArrayList<String> arraylist2 = method(array,
                str -> str.split(",")[1].equals("女"),
                str -> str.split(",")[0].length() == 4);

        System.out.println(arraylist2);
        for (String s : arraylist2) {
            System.out.println(s);
        }


Function接口【转换数据类型的接口】

java.util.function.Function<T,R>接口用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,后者称为后置条件

抽象方法:apply

Function接口中最主要的抽象方法为:R apply(T t),根据类型T的参数获取类型R的结果。
使用的场景例如∶将String 类型转换为Integer类型。

import java.util.function.Function;
/*
    java.util .function. Function<T.R>接口用来根据一个类型的激据得到另一个类型的数据,
        前者称为前置条件,后者称为后置条件。
    Function接口中最主要的抽象方法为: R apply(T t),根据类型T的参数获取类型R的结果。
    使用的场景例如:将string类型转换为Integer类型。
 */
public class DemosFunctionApply {
    /*
        定义一个方法
            方法的参数传递一个字符串类型的整数
            方法的参数传递一个Function接口,泛型使用<string,Integer>
            使用Function接口中的方法apply,把字符串类型的整数转换为Integer类型的整数
     */
    private static void method (String str, Function<String,Integer> fun){
        Integer num = fun.apply(str);
        System.out.println(num);
    }
    public static void main(String[] args){
        String s = "1234";
        method(s,(str)->{
            return Integer.parseInt(str);
        });

        method("9845783",(str)->{
            return Integer.parseInt(str);
        });
        //Lambda表达式简化
        method(s,str-> Integer.parseInt(str));
    }
}


默认方法:andThen

Function 接口中有一个默认的andThen方法,用来进行组合操作。JDK源代码如︰

default <v> Function<T,V> andThen(Function<? super R,? extends v> after){
        Objects.requireNonNull(after);
        return (T t) -> after.apply(apply(t));
}

该方法同样用于"先做什么,再做什么""的场景,和Consumer中的 andThen差不多:

import java.util.function.Function;
public class Demo12FunctionAndThen {
    private static void method(Function<String,Integer>one,Function<Integer,Integer> two){
        int num = one.andThen(two ).apply("10");
        system.out.println(num + 20);
    }
    public static void main( String[] args){
        method(Integer::parseInt, i->i*=10);
    }
}

第一个操作是将字符串解析成为int数字,第二个操作是乘以10。两个操作通过 andThen按照前后顺序组合到了一起。

import java.util.function.Function;
/*
    Function接口中的黑认方法andThen:用来进行组合操作

    需求:
        把string类型的"123”",转换为Inteter类型,把转换后的结果加10
        把增加之后的iInteger类型的数据,转换为string类型

    分析:
        转换了两次
        第一次是把string类型转换为了Integer类型
            所以我们可以使用Function<string, Integer> fun1
                Integer i = fun1.apply ("123")+10;
        第二次是把Integer类型转换为string类型
            所以我们可以使用Function<Integer,string> fun2
                string s = fun2.apply(i);
        我们可以使用andThen方法,把两次转换组合在一起使用
            string s = fun1.andThen(fun2 ). apply ( "123" );
            fun1先调用apply方法,把字符串转换为Integer
            fun2再调用apply方法,把Integer转换为字符串
 */
public class DemosFunctionApply {
    /*
        定义一个方法
            参数串一个字符串类型的整数
            参数再传递两个Function接口
                一个泛型使用Function<string , Integer>
                一个泛型使用Function<Integer ,string>
     */
    private static void method1(String str, Function<String,Integer> fun1,Function<Integer,String> fun2){
        Integer num1 = fun1.apply(str)+10;
        String num2 = fun2.apply(num1);
        System.out.println(num2);
    }

    private static void method2(String str, Function<String,Integer> fun1,Function<Integer,String> fun2){
        String num3 = fun1.andThen(fun2).apply(str);//使用addThen方法拼接fun1和fun2,一起apply
        System.out.println(num3);
    }
    public static void main(String[] args){
        String s = "1234";
        method1(s,(str)->{
            /*把字符串转换成整数*/
            return Integer.parseInt(str);
        },(integer)->{
            /*把整数转换成字符串*/
            return integer+"";
        });
        //简化Lambda表达式
        method1(s,/*把字符串转换成整数*/Integer::parseInt,/*双冒号用来简化Lambda表达式,连参数和箭头都不用写*/
                integer->/*把整数转换成字符串*/integer+"");

        method2(s, str->Integer.parseInt(str)+10,integer->integer+"");
    }
}    


练习题:自定义函数模型拼接
题目

请使用Function进行函数模型的拼接,按照顺序需要执行的多个函数操作为︰
String str = "赵丽颖,20";

  1. 将字符串截取数字年龄部分,得到字符串;
  2. 将上一步的字符串转换成为int类型的数字;
  3. 将上一步的int数字累加100,得到结果int数字。
import java.util.function.Function;

public class DemosDiyFunction {
    private static void method(String str, Function<String,String> fun1, Function<String,Integer> fun2, Function<Integer,Integer> fun3){
//        String f1 = fun1.apply(str);
//        Integer f2 = fun2.apply(f1);
//        int f3 = fun3.apply(f2);
        Integer num = fun1.andThen(fun2).andThen(fun3).apply(str);//使用addThen方法拼接fun1和fun2和fun3,一起apply
        System.out.println(num);
    }
    public static void main(String[] args){
        String s = "娜可露露,17";
        method(s,(str)->{
            return s.split(",")[1];
        },(String str)->{
            return Integer.parseInt(str);
        },(i)->{
            /*把整数转换成字符串*/
            return i+100;
        });
        //简化代码
        method(s,str->s.split(",")[1],Integer::parseInt,i->i+100);
    }
}


Stream流

说到Stream便容易想到I/O Stream,而实际上,谁规定"流"就一定是"I0流"呢?在Java8中,得益于Lambda所带来的函数式编程,引入了一个全新的Stream概念用于解决已有集合类库既有的弊端

引言

传统集合的多步遍历代码

几乎所有的集合(如Collection 接口或Map接口等)都支持直接或间接的遍历操作。而当我们需要对集合中的元素进行操作的时候,除了必需的添加、删除、获取外,最典型的就是集合遍历。例如∶

import java.util.ArrayList;
import java.util.List;
/*
    使用传统的方式,遍历集合,对集合中的数据进行过滤
 */
public class DemosDatafilter{
    public static void main(String[] args) {

//创建一个List集合,存储姓名
        List<String> list = new ArrayList<>();
        list.add("张无忌");
        list.add("周芷若");
        list.add("赵敏");
        list.add("张强");
        list.add("张三丰");
//对list集合中的元素进行过滤,只要以张开头的元素,存储到一个新的集合中
        List<String> listA = new ArrayList<>();
        for (String s : list) {
            //如果数据以"张"字开头,就把数据添加进listA集合中
            if (s.startsWith("张")){
                listA.add(s);
            }
        }

        //对listA集合进行过滤,只要姓名长度为3的人,存储到一个新集合中
        List<String> listB = new ArrayList<>();
        for (String s : listA) {
            if (s.length()==3){
                listB.add(s);
            }
        }
        //遍历listB集合
        for (String s : listB) {
            System.out.println(s);
        }
    }
}


使用Stream流的方式遍历集合

上面的传统代码中含有三个循环,每一个作用不同:

  1. 首先筛选所有姓张的人;
  2. 然后筛选名字有三个字的人;
  3. 最后进行对结果进行打印输出。

每当我们需要对集合中的元素进行操作的时候,总是需要进行循环、循环、再循环。这是理所当然的么?不是。循环是做事情的方式,而不是目的。另一方面,使用线性循环就意味着只能遍历一次。如果希望再次遍历,只能再使用另一个循环从头开始。

那,Lambda的衍生物Stream能给我们带来怎样更加优雅的写法呢?答案是:有

import java.util.ArrayList;
import java.util.List;

/*
    使用stream流的方式,遍历集合,对集合中的数据进行过滤
    stream流是JDK1.8之后出现的
    关注的是做什么,而不是怎么做
 */
public class DemosDatafilterStream {
    public static void main(String[] args) {

//创建一个List集合,存储姓名
        List<String> list = new ArrayList<>();
        list.add("张无忌");
        list.add("周芷若");
        list.add("赵敏");
        list.add("张强");
        list.add("张三丰");
//对list集合中的元素进行过滤,只要以张开头的元素,存储到一个新的集合中
//如果数据以"张"字开头,就把数据添加进listA集合中
//对listA集合进行过滤,只要姓名长度为3的人,存储到一个新集合中
// 遍历listB集合
        list.stream().filter((String name)->{
            return name.startsWith("张");
        }).filter((String name)->{
            return name.length()==3;
        }).forEach((String name)->{
            System.out.println(name);
        });

        //代码简化
        list.stream()
                .filter(name-> name.startsWith("张"))
                .filter(name-> name.length()==3)
                .forEach(name-> System.out.println(name));
    }
}


流式思想概述

注意:请暂时忘记对传统IO流的固有印象!
整体来看,流式思想类似于工厂车间的"生产流水线"。

当需要对多个元素进行操作(特别是多步操作)的时候,考虑到性能及便利性,我们应该首先拼好一个"模型"步骤方案,然后再按照方案去执行它。

这张图中展示了过滤、映射、跳过、计数等多步操作,这是一种集合元素的处理方案,而方案就是一种"函数模型”。图中的每一个方框都是一个"流”,调用指定的方法,可以从一个流模型转换为另一个流模型。而最右侧的数字3是最终结果。

这里的filter 、map 、skip都是在对函数模型进行操作,集合元素并没有真正被处理。只有当终结方法count执行的时候,整个模型才会按照指定策略执行操作。而这得益于Lambda的延迟执行特性。

备注:"Stream流"其实是一个集合元素的函数模型,它并不是集合,也不是数据结构,其本身并不存储任何元素(或其地址值)。

个人理解:这样的好处是,如果我们不执行最后的终结方法count中间的3步也不会执行,这样可以提高程序效率

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

  • 元素是特定类型的对象,形成一个队列。Java中的Stream并不会存储元素,而是按需计算。
  • 数据源流的来源。可以是集合,数组等。

和以前的Collection操作不同,Stream操作还有两个基础的特征:

  • Pipelining:中间操作都会返回流对象本身。这样多个操作可以串联成一个管道,如同流式风格(fluentstyle )。这样做可以对操作进行优化,比如延迟执行(laziness)和短路( short-circuiting)。
  • 内部迭代∶以前对集合遍历都是通过lterator或者增强for的方式,显式的在集合外部进行迭代,这叫做外部迭代。Stream提供了内部迭代的方式,流可以直接调用遍历方法。

当使用一个流的时候,通常包括三个基本步骤∶获取一个数据源( source )→数据转换→执行操作获取想要的结果,每次转换原有Stream对象不改变,返回一个新的Stream对象(可以有多次转换),这就允许对其操作可以像链条一样排列,变成一个管道。


两种获取Stream流的方式

java.util.stream.Stream是Java 8新加入的最常用的流接口。(这并不是一个函数式接口。)

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

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

根据Collection获取流

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

import java.util.*;
import java.util.stream.Stream;

/*
    java.util.stream.Stream<T>是Java 8新加入的最常用的流接口。(这并不是一个还能输式接口。)
    获取一个流非常简单,有以下几种常用的方式:
        -所有的ColLectiorn集合都可以通过stream认方法获取流,
            default streom<E>stream .
        - streaw接口的静态方法of可以获取数组对应的流。
            static <T> Stream<T> of (T... values)
            参数是一个可变参数,那么我们就可以传递一个数组
 */
public class DemosGetStream {
    public static void main(String[] args) {
        //把集合转换为Stream流
        List<String> list = new ArrayList<>();
        Stream<String> stream1 = list.stream();

        Set<String> set = new HashSet<>();
        Stream<String> stream2 = set.stream();

        Map<String, String> map = new HashMap<>();
        //获取键,把键存储到一个set集合中
        Set<String> keyset = map.keySet();
        Stream<String> stream3 = keyset.stream();
        //获取值,存储到一个Collection集合中
        Collection<String> values = map.values();
        Stream<String> stream4 = values.stream();
        //获取键值对
        Set<Map.Entry<String, String>> entries = map.entrySet();
        Stream<Map.Entry<String, String>> stream5 = entries.stream();
        
        //把数组转换为Stream流
        Stream<Integer> stream6 = Stream.of(1, 2, 3, 4, 5);
        //可变参数可以传递子数组
        Integer[] arr1 = {1,2,3,4,5};
        Stream<Integer> stream7 = Stream.of(arr1);
        String[] arr2 = {"a","af","e","b","c"};
        Stream<String> stream8 = Stream.of(arr2);

    }
}

Stream流中的常用方法

流模型的操作很丰富,这里介绍一些常用的API。这些方法可以被分成两种︰

  • 延迟方法:返回值类型仍然是 Stream接口自身类型的方法,因此支持链式调用。(除了终结方法外,其余方法均为延迟方法。)
  • 终结方法:返回值类型不再是Stream接口自身类型的方法,因此不再支持类似 StringBuilder那样的链式调用。本小节中,终结方法包括count和forEach方法。

备注:本小节之外的更多方法,请自行参考API文档。

逐一处理:forEach

虽然方法名字叫forEach,但是与for循环中的"for-each"昵称不同。

void forEach(Consumer<? super T action) ;

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

复习consumer接口
java.util.function.consumer<T>接口是一个消费型接口。
Consumer接口中包含抽象方法void accept(T t),意为消费一个指定泛型的数据。
基本使用:
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));
	}
}
import java.util.stream.Stream;

/*
    Stream流中的常用方法_forEach
    void forEach(Consumer<? super T> action ) ;
    该方法接收一个Consumer接口函数,会将每一个流元素交给该函数进行处理。
    Consumer接口是一个消费型的函数式接口,可以传递Lambda表达式,消费数据

    简单记:
        forEoch方法,用来遍历流中的数据
        是一个终结方法,遍历之后就不能唯续调用streaw流中的其他方法


    )
 */
public class DemosStream_forEach {
    public static void main(String[] args) {
        //获取一个stream流
        Stream<String> stream = Stream.of("张三","李四","王五","赵六","田七");
        //使用Stream流中的方法forEach对Stream流中的数据进行遍历
//        stream.forEach((String s)->{
//            System.out.println(s);
//        });//执行完forEach后,流就会被关闭
        System.out.println("========================================");
        //代码优化
        stream.forEach( s-> System.out.println(s));
    }
}


过滤:filter

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

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

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

复习Predicate接口

此前我们已经学习过java.util.stream.Predicate函数式接口,其中唯一的抽象方法为︰

boolean test(T t);

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

基本使用
import java.util.stream.Stream;

/*
    streow流中的常用方法_fiLter:用于对Stream流中的数据进行过滤
    Stream<T>. filter (Predicate<? super T> predicate);
    filter方法的参数Predicate是一个函数式接口,所以可以传递Lambda表达式,对数据进行过滤
    Predicate中的抽象方法:
        booLean test(T t);
 */
public class DemosStream_filter {
    public static void main(String[] args) {
        //创建一个Stream流
        Stream<String> stream = Stream.of("张三丰","张翠山","杨不悔","赵敏","张无忌","周芷若");
        /*//对Stream流中的元素进行过滤,只要姓"张"的人
        Stream<String> stream2 = stream.filter((String str) -> {
            return str.startsWith("张");
        });
        //遍历stream2这个流
        stream2.forEach((String str)->{
            System.out.println(str);
        });*/

        System.out.println("=====================");

        //代码优化
        //对Stream流中的元素进行过滤,只要姓"张"的人
        Stream<String> stream3 = stream.filter(str -> str.startsWith("张"));
        //遍历stream2这个流
        stream3.forEach(str-> System.out.println(str));
    }

}


注意事项:Stream流的特点【只能使用一次】
Stream流属于管道流,只能被消费(使用)一次
第一个Stream流调用完毕方法之后,数据就会流转到下一个Stream流上
而此时第一个Stream流已经使用完毕了,就会关闭了
所以第一个Stream流就不能再调用方法了
映射: 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类型,而这种转换的动作,就称为"映射"。

基本使用
import java.util.stream.Stream;

/*
    如果需要将流中的元素映射到另一个流中,可以使用map方法。
    方法签名︰
        <R> Stream<R> map(Function<? super T, ? extends R> mapper);
    该接口需要一个Function函数式接口参数,可以将当前流中的T类型数据转换为另一种R类型的流。
    Function中的抽象方法:
        R apply(T t);
 */
public class DemosStream_map {
    public static void main(String[] args) {
        //获取一个String类型的Stream流
        Stream<String> stream1 = Stream.of("1", "2", "3", "4", "5");
        //使用map方法,吧字符串类型的整数,转换(映射)为Integer类型的整数
        Stream<Integer> stream2 = stream1.map(str -> Integer.parseInt(str));
        //遍历
        stream2.forEach(i-> System.out.println(i));
    }
}

这段代码中,map方法的参数通过方法引用,将字符串类型转换成为了int类型(并自动装箱为Integer类对象


统计个数:count

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

long count();

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

基本使用:
import java.util.stream. Stream;

public class DemosStreamCount{
	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
	}
}
import java.util.ArrayList;
import java.util.stream.Stream;

/*
    streom流中的常用方法_count:用于统计stream流中元素的个数
    Long count( );
    count方法足个终结方法,返回值是一个Long类型的整数
    所以不能再继续调用Stream流中的其他方法了
 */
public class DemosStream_count {
    public static void main(String[] args) {
        //获取一个Stream流
        ArrayList<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        list.add(4);
        list.add(5);
        list.add(6);
        list.add(7);
        Stream<Integer> stream = list.stream();
        long count = stream.count();
        System.out.println(count);//7
    }
}

取用前几个:limit

limit方法可以对流进行截取,只取用前n个。方法签名︰

Stream<T> limit(long maxSize);
基本使用:
import java.util.ArrayList;
import java.util.stream.Stream;

/*
    Stream流中的常用方法limit:用于截取流中的元素
    limit方法可以对流进行取,只取用前n个。方法签名:
    streom<TLimit (Long moxSize);
    参数是一个long型,如果集合当前长度大于参数则进行截取;否则不进行操作
    Limit方法是一个延迟方法,只是对流中的元索进行截取,返回的是一个新的流,所以可以继续调用Stream流中的其他方法
 */
public class DemosStream_limit {
    public static void main(String[] args) {
        //获取一个Stream流
        String[] arr = {"美羊羊","喜洋洋","懒洋洋","灰太狼","红太狼"};
        Stream<String> stream1 = Stream.of(arr);
        //取前3个元素
        Stream<String> stream2 = stream1.limit(3);
        //遍历
        stream2.forEach(s-> System.out.println(s));
    }
}


跳过前几个: skip

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

Stream<T> skip(long n);

如果流的当前长度大于n,则跳过前n个∶否则将会得到一个长度为0的空流。

基本使用:

import java.util.stream.Stream;

/*
    stream流中的常用方法_skip:用于跳过元素
    加如果希望跳过前几个元素,可以使用skip方法获取一个截取之后的新流:
    Stream<T> Skip(long n);
        如果流的当前长度大于n,则跳过前n个;否则将会得到一个长度为0的空流。
 */
public class DemosStream_skip {
    public static void main(String[] args) {
        //获取一个Stream流
        String[] arr = {"美羊羊","喜洋洋","懒洋洋","灰太狼","红太狼"};
        Stream<String> stream1 = Stream.of(arr);
        //跳过前3个元素
        Stream<String> stream2 = stream1.skip(3);
        //遍历
        stream2.forEach(s-> System.out.println(s));
    }
}


组合:concat

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

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

备注:这是一个静态方法,与java.lang . string当中的concat方法是不同的。

基本使用:
import java.util.stream.Stream;

/*
    stream流中的常用方法_skip:用于跳过元素  个截取之后的新流:
    Stream<T> Skip(long n);
        如果流的当前长度大于n,则跳过前n个;否则将会得到一个长度为0的空流。
 */
public class DemosStream_concat {
    public static void main(String[] args) { 
        //创建一个Stream流
        String[] arr = {"美羊羊","喜洋洋","懒洋洋","灰太狼","红太狼"};
        Stream<String> stream2 = Stream.of(arr);
        //创建一个Stream流
        Stream<String> stream = Stream.of("张三丰","张翠山","杨不悔","赵敏","张无忌","周芷若");
        //组合上面两个流
        Stream<String> concat = Stream.concat(stream, stream2);
        //遍历
        concat.forEach(s-> System.out.println(s));
    }
}


练习1:集合元素的处理(传统方式
题目

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

  1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
  2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
  3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
  4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
  5. 将两个队伍合并为一个队伍;存储到一个新集合中。
  6. 根据姓名创建 Person对象;存储到一个新集合中。
  7. 打印整个队伍的Person对象信息。
package MyTest;
import java.util.ArrayList;
import java.util.List;
/*
    现在有两个ArrayList集合存储队伍当中的多个成员姓名,要求使用传统的for循环(或增强for循环)

    依次进行以下若干操作步骤∶
        1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
        2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
        3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
        4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
        5. 将两个队伍合并为一个队伍;存储到一个新集合中。
        6. 根据姓名创建 Person对象;存储到一个新集合中。
        7. 打印整个队伍的Person对象信息。
 */
public class DemosTest1 {
    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("洪七公");

        //1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
        ArrayList<String> one1 = new ArrayList<>();
        for (String s : one) {
            if (s.length()==3){
                one1.add(s);
            }
        }
        System.out.println(one1);
        //2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
        ArrayList<String> one2 = new ArrayList<>();
//        one2.add(one1.get(0));
//        one2.add(one1.get(1));
//        one2.add(one1.get(2));
        for (int i = 0; i < 3; i++) {
            one2.add(one1.get(i));
        }
        System.out.println(one2);


        //第二支队伍
        ArrayList<String> two = new ArrayList<>();
        two.add("古力娜扎");
        two.add("张无忌");
        two.add("赵丽颖");
        two.add("张三丰");
        two.add("尼古拉斯赵四");
        two.add("张天爱");
        two.add("张二狗");

        //3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
        ArrayList<String> two1 = new ArrayList<>();
        for (String s : two) {
            if (s.startsWith("张")){
                two1.add(s);
            }
        }
        System.out.println(two1);
        //4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
        ArrayList<String> two2 = new ArrayList<>();
        for (int i = 2; i < two1.size(); i++) {
            two2.add(two1.get(i));
        }
        System.out.println(two2);
        //5. 将两个队伍合并为一个队伍;存储到一个新集合中。
        ArrayList<String> three = new ArrayList<>();
        for (String s : one2) {
            three.add(s);
        }
        for (String s : two2) {
            three.add(s);
        }
        System.out.println(three);
        //6. 根据姓名创建 Person对象;存储到一个新集合中。
        ArrayList<Person> four = new ArrayList<>();
        for (String s : three) {
            four.add(new Person(s));
        }
        //7. 打印整个队伍的Person对象信息。
        for (Person person : four) {
            System.out.println(person.getName());
        }
    }
}


练习2:集合元素的处理(Stream流的方式
题目

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

  1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
  2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
  3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
  4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
  5. 将两个队伍合并为一个队伍;存储到一个新集合中。
  6. 根据姓名创建 Person对象;存储到一个新集合中。
  7. 打印整个队伍的Person对象信息。
package MyTest;
import java.util.ArrayList;
import java.util.stream.Stream;

/*
    现在有两个ArrayList集合存储队伍当中的多个成员姓名,要求使用传统的for循环(或增强for循环)

    依次进行以下若干操作步骤∶
        1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
        2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
        3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
        4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
        5. 将两个队伍合并为一个队伍;存储到一个新集合中。
        6. 根据姓名创建 Person对象;存储到一个新集合中。
        7. 打印整个队伍的Person对象信息。
 */
public class DemosTest2 {
    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("洪七公");

        //1. 第一个队伍只要名字为3个字的成员姓名﹔存储到一个新集合中。
        Stream<String> oneStream = one.stream().filter((s) -> s.length() == 3);
        //2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
        Stream<String> oneStream1 = oneStream.limit(3);

        //第二支队伍
        ArrayList<String> two = new ArrayList<>();
        two.add("古力娜扎");
        two.add("张无忌");
        two.add("赵丽颖");
        two.add("张三丰");
        two.add("尼古拉斯赵四");
        two.add("张天爱");
        two.add("张二狗");

        //3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
        Stream<String> twoStream = two.stream().filter(s -> s.startsWith("张"));
        //4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
        Stream<String> twoStream1 = twoStream.skip(2);
        //5. 将两个队伍合并为一个队伍;存储到一个新集合中。
        Stream<String> concatStream = Stream.concat(oneStream1, twoStream1);
        //6. 根据姓名创建 Person对象;存储到一个新集合中。
        Stream<Person> personStream = concatStream.map(s -> new Person(s));
        //7. 打印整个队伍的Person对象信息。
        personStream.forEach(name-> System.out.println(name));
    }
}


方法引用

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

方法引用符

双冒号::为引用运算符,而它所在的表达式被称为方法引用。如果Lambda要表达的函数方案已经存在于某个方法的实现中,那么则可以通过双冒号来引用该方法作为Lambda的替代者。

package MethodReference;

public class DemosPrintable {
    //定义一个方法,参数传递Printable接口,对字符串进行打印
    public static void printString(Printable p){
        p.print("hello Java!");
    }

    public static void main(String[] args){
        //调用printString方法,方法的参数Printable是一个函数式接口,所以可以传递Lambda表达式
        printString(s->System.out.println(s));
        /*
            分析:
                Lombda表达式的目的,打印参数传递的字符串
                把参数s,传递给了system.out对象,调用out对象中的方法println对字符串进行了输出
                注意:
                    1.System.out对象是已经存在的
                    2.println方法也是已经存在的
                所以我们可以使用方法引用来优化Lambda表达式
         */
        printString(System.out::println);
    }
}

语义分析

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

  • Lambda表达式写法:s -> system.out.println(5);
  • 方法引用写法:` system.outI : println

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

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

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

推导与省略

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

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


通过对象名引用成员方法

这是最常用的一种用法,与上例相同。如果一个类中已经存在了一个成员方法

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类的对象实例,则可以通过对象名引用成员方法,代码为︰

package MethodReference;
/*
    通过对象名引用成员方法
    使用前提是对象名是已经存在的,成员方法也是已经存在的
    就可以使用对象名引用成员方法
 */
public class DemosObjectMethodReference {
    //定义一个方法,方法的参数传递Printable接口
    public static void pringString(Printable p){
        p.print("Java!");
    }

    public static void main(String[] args) {
        //调用printString方法,方法的参数Printable是一个函数式接口,所以可以传递Lambda表达式
        pringString((s)->{
            //创建MethodReferanceObject对象
            MethodReferenceObject object = new MethodReferenceObject();
            //调用MethodReferanceObject对象中的成员方法printUpperCaseString,把字符串按照大写输出
            object.printUpperCaseString(s);
        });

        /*
            使用方法引用优化Lambda
            对象是已经存在的-----MethodReferanceObject对象
            成员方法也是已经存在的printUpperCaseString
            所以我们可以使用对象名引用成员方法
         */
        MethodReferenceObject object = new MethodReferenceObject();
        pringString(object::printUpperCaseString);
    }
}


通过类名引用静态方法

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

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

第—种写法是使用Lambda表达式:

public class DemosLambda {
	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));
    }
}

但是使用方法引用的更好写法是:

package ClassReferenceStaticMethod;

import java.lang.reflect.Method;

public class DemosClassReferenceStaticMethod {
    //定义一个方法,方法的参数传递Calcable这个函数式接口和要传入的负整数
    public static int method(Calcable c,int num){
        return c.calsAbs(num);
    }

    public static void main(String[] args) {
        int i = method((a) -> {
            return Math.abs(a);
        }, -100);
        System.out.println(i);

        /*
            【类名引用静态方法】ClassReferenceStaticMethod
            使用方法引用优化Lambda表达式
            前提:
            Math类是存在的
            abs计算绝对值的静态方法也是已经存在的
            所以我们可以直接通过类名引用静态方法
         */
        int i1 = method(Math::abs, -100);
        System.out.println(i1);
    }
}


通过super引用成员方法

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

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

然后是父类 Human的内容︰

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

最后是子类 Man的内容,其中使用了Lambda的写法︰

package SuperReference;

public class Man extends Human{
    @Override
    public void sayHello() {
        System.out.println("大家好,我是Man!");
    }

    public void method(Greetable g){
        g.greet();
    }

    public void show(){
        //调用method方法,方法的参数Greetable是一个函数式接口,所以可以传递Lambda表达式
        method(()->{
            //创建父类Human对象
            Human h = new Human();
            //调用父类的sayHello方法
            h.sayHello();
        });

        //因为有子父类关系,所以存在一个关键字super,代表父类,所以我们可以直接使用super调用父类的成员方法
        method(()->{
            super.sayHello();
        });
        //因为有子父类关系,所以存在一个关键字this,代表子类,所以我们可以直接使用this调用子类的成员方法
        method(()->{
            this.sayHello();
        });


        /*  优化上面的Lambda
            使用super引用类的成员方法
            前提:
                super是已经存在的
                父类的成员方法sayHeLLo也是已经存在的
                所以我们可以直接使用super引用灾类的成员方法
         */
        method(super::sayHello);
    }

    public static void main(String[] args) {
        new Man().show();
    }
}


通过this引用本类成员方法

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

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

然后是父类 Human的内容︰

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

最后是子类 Man的内容,其中使用了Lambda的写法︰

package SuperReference;

public class Man extends Human{
    @Override
    public void sayHello() {
        System.out.println("大家好,我是Man!");
    }

    public void method(Greetable g){
        g.greet();
    }

    public void show(){
        //调用method方法,方法的参数Greetable是一个函数式接口,所以可以传递Lambda表达式
        method(()->{
            //创建父类Human对象
            Human h = new Human();
            //调用父类的sayHello方法
            h.sayHello();
        });

        //因为有子父类关系,所以存在一个关键字super,代表父类,所以我们可以直接使用super调用父类的成员方法
        method(()->{
            super.sayHello();
        });
        //因为有子父类关系,所以存在一个关键字this,代表本类,所以我们可以直接使用this调用子类的成员方法
        method(()->{
            this.sayHello();
        });


        /*  优化上面的Lambda
            使用super引用类的成员方法
            前提:
                super是已经存在的
                父类的成员方法sayHeLLo也是已经存在的
                所以我们可以直接使用super引用灾类的成员方法
         */
        method(super::sayHello);

        /*  优化上面的Lambda
            使用方法引用优化Lambda表达式
            前提
            this是已经存在的
            本类的成员方法buyHouse也是已经存在的
            所以我们可以直接使用this引用本类的成员方法sayHello
         */
        method(this::sayHello);
    }

    public static void main(String[] args) {
        new Man().show();
    }
}


类的构造器引用

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

package MyTest;

public class Person {
    private String name;

    public Person() {
    }

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

    public String getName() {
        return name;
    }

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

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

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

package MyTest;
@FunctionalInterface
public interface PersonBuilder {
    //定义一个方法,根据传递的姓名,创建Person对象返回
    public abstract Person buliderPerson(String name);
}
package MyTest;


/*
    类的构造器(构造方法)引用
 */
public class DemosTest {
    //定义一个方法,参数传递姓名和PersonBuilder接口,方法中通过姓名创建Person对象
    public static void printName(String name,PersonBuilder pb){
        Person person = pb.buliderPerson(name);
        System.out.println(person.getName());
    }

    public static void main(String[] args) {
        //调用printName方法,方法的参数PersonBuilder接口是一个函数式接口,所以可以传递一个Lambda表达式
        printName("牛牛牛",(s)->{
            return new Person(s);
        });

        /*
            【类的构造器(构造方法)引用】
            使用方法引用优化Lambda表达式
            前提:
            构造方法new Person(String name)已知
            创建对象已知
            就可以使用Person引用new创建对象
         */
        printName("牛牛牛",Person::new);
    }

}


数组的构造器引用

也就是使用方法引用来创建一个数组

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

@FunctionalInterface
public interface ArrayBuilder{
	int[] buildArray(int length);
}

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

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

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

import java.util.Arrays;

/*
    数组的构造器引用
*/
public class Demo {
    /*
        定义一个方法
        方法的参数传读创建数组的长度和ArrayBuilder接口
        方法内部根据传递的长度使用ArrayBuilder中的方法创建数组并返回
    */
    public static int[] createArray(int length,ArrayBuilder ab){
        return ab.buildArray(length);
    }

    public static void main(String[] args){
        //调用createArray方法,传递数组的长度和iambda表达式
        int[]arr1 = createArray( 10, (len)-> {
            //根据数组的长度,创建数组并返回
            return new int[len];
        });
        System.out.println(arr1.length);

        /*
            使用方法引用优化Lambda表达式
            已知创建的颇是int[]数组
            数组的长度也是已知的
            就可以使用方法引用
            int[]引用new,根据参数传递的长度来创建数组
         */
        int[] arr2 = createArray(10,int[]::new);
        System.out.println(Arrays.toString(arr2));
        System.out.println(arr2.length);//10
    }
}


posted @ 2021-05-12 18:24  牛牛ō^ō  阅读(331)  评论(0编辑  收藏  举报