jdk8新特性之双冒号 :: 用法及详解

jdk8的新特性有不少,最亮眼的当属函数式编程的语法糖,本文主要讲解下双冒号::的用法。java

概念

类名::方法名,至关于对这个方法闭包的引用,相似js中的一个function。好比:编程

Function<String,String> func =  String::toUpperCase;

 

(Function在java.util.function包下,也是jdk8新加入的类,同级目录下有不少函数式编程模型接口,好比Consumer/Predicate/Operator等)闭包

 

func至关于一个入参和出参都为String的函数,能够直接app

func.apply("abc")

 

接收一个参数,返回一个结果("ABC")。也能够用于代替下面的Lambda表达式:jvm

List<String> l = Arrays.asList("a","b","c");
l.stream().map(s -> s.toUpperCase());
l.stream().map(func);

 

下面自定义一个函数式接口ide

public class MyConsumer<String> implements Consumer<String> {
    @Override
    public void accept(String s) {
        System.out.println(s);
    }
}

 

下面这俩种写法等价:函数式编程

List<String> l = Arrays.asList("a","b","c");

l.forEach(new MyConsumer<>());
l.forEach(s -> System.out.println(s));

 

可是,这种写法却不行,编译失败:函数

l.forEach(MyConsumer::accept);

 

由于MyConsumer的accept方法不是静态的,若是想使用这个方法,须要一个实例,还须要一个入参,共俩个参数。而List.forEach中须要的是consumer类型,至关于s -> {...},只有一个参数。this

 

下面详细分析双冒号使用的各类状况

新建一个类,里面声明四个表明各类状况的方法:code

public class DoubleColon {

    public static void printStr(String str) {
        System.out.println("printStr : " + str);
    }

    public void toUpper(){
        System.out.println("toUpper : " + this.toString());
    }

    public void toLower(String str){
        System.out.println("toLower : " + str);
    }

    public int toInt(String str){
        System.out.println("toInt : " + str);
        return 1;
    }
}

 

把它们用::提取为函数,再使用:

Consumer<String> printStrConsumer = DoubleColon::printStr;
printStrConsumer.accept("printStrConsumer");

Consumer<DoubleColon> toUpperConsumer = DoubleColon::toUpper;
toUpperConsumer.accept(new DoubleColon());

BiConsumer<DoubleColon,String> toLowerConsumer = DoubleColon::toLower;
toLowerConsumer.accept(new DoubleColon(),"toLowerConsumer");

BiFunction<DoubleColon,String,Integer> toIntFunction = DoubleColon::toInt;
int i = toIntFunction.apply(new DoubleColon(),"toInt");

 

非静态方法的第一个参数为被调用的对象,后面是入参。静态方法由于jvm已有对象,直接接收入参。

 

再写一个方法使用提取出来的函数:

public class TestBiConsumer {
    public void test(BiConsumer<DoubleColon,String> consumer){
        System.out.println("do something ...");
    }
}

 

下面这俩种传入的函数是同样的:

TestBiConsumer obj = new TestBiConsumer();
obj.test((x,y) -> System.out.println("do something ..."));
obj.test(DoubleColon::toLower);

 

总结

用::提取的函数,最主要的区别在于静态与非静态方法,非静态方法比静态方法多一个参数,就是被调用的实例。

相关文章
相关标签/搜索