转自:详解JAVA8函数式接口
转自:Java8之Consumer、Supplier、Predicate和Function攻略
目录
概念
函数式接口在java中是指:有且仅有一个抽象方法的接口。
函数式接口,即适用于函数式编程场景的接口。而java中的函数式编程体现就是Lambda,所以函数式接口就是可以适用于Lambda使用的接口。只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利地进行推导。
备注:“语法糖"是指使用更加方便,但是原理不变的代码语法。例如在遍历集合时使用的for-each语法,其实底层的实现原理仍然是迭代器,这便是“语法糖”。从应用层面来讲,Java中的Lambda可以被当做是匿名内部类的“语法糖”,但是二者在原理上是不同的。
实现
案例1:
step1:首先申明一个函数式接口。(注意:实际开发中public abstract 可以不写)
@FunctionalInterface
public interface MyFunctionInterface {
//public abstract void mrthod(String str); //抽象方法
void mrthod(String str); //抽象方法
}
一旦使用该注解来定义接口,编译器将会强制检查该接口是否确实有且仅有一个抽象方法,否则将会报错。需要注意的是,即使不使用该注解,只要满足函数式接口的定义,这仍然是一个函数式接口,使用起来都一样。(该接口是一个标记接口)。
step2:在另一个方法中作为参数传入,并且里面写调用的函数式接口中的方法。
// 定义一个含有函数式接口的方法
public static void doSomthing(MyFunctionInterface functionalInterface) {
......
functionalInterface.mrthod("hello world");
}
step3:实现函数式接口中的抽象方法。
doSomthing((str) -> {
System.out.println("这是我传入的参数"+str);
});
案例2:
step1:
@FunctionalInterface
public interface MyFunction<A,B,C> {
C Do(A a,B b);
}
step2:
public static void main(String[] args){
MyFunction fun = (a,b) -> (int) a+ (int)b;
System.out.println(fun.Do(1,2));
}
1、Consumer接口
consumer接口
就是一个消费型的接口,通过传入参数,然后输出值,就是这么简单,Java8 的一些方法看起来很抽象,其实,只要你理解了就觉得很好用,并且非常的简单。
//① 使用consumer接口实现方法
Consumer<String> consumer = new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
};
Stream<String> stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
stream.forEach(consumer);
System.out.println("********************");
//② 使用lambda表达式,forEach方法需要的就是一个Consumer接口
stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
Consumer<String> consumer1 = (s) -> System.out.println(s);//lambda表达式返回的就是一个Consumer接口
stream.forEach(consumer1);
//更直接的方式
//stream.forEach((s) -> System.out.println(s));
System.out.println("********************");
//③ 使用方法引用,方法引用也是一个consumer
stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
Consumer consumer2 = System.out::println;
stream.forEach(consumer);
结果:
分析:
① consumer
接口分析
在代码①中,我们直接创建 Consumer
接口,并且实现了一个名为 accept
的方法,这个方法就是这个接口的关键了。
我们看一下 accept
方法;这个方法传入一个参数,不返回值。当我们发现 forEach
需要一个 Consumer
类型的参数的时候,传入之后,就可以输出对应的值了。
② lambda 表达式作为 consumer
Consumer<String> consumer1 = (s) -> System.out.println(s);//lambda表达式返回的就是一个Consumer接口
在上面的代码中,我们使用下面的 lambda
表达式作为 Consumer
。仔细的看一下你会发现,lambda
表达式返回值就是一个 Consumer
;所以,你也就能够理解为什么 forEach
方法可以使用 lamdda 表达式作为参数了吧。
③ 方法引用作为 consumer
Consumer consumer2 = System.out::println;
总结:
看完上面的实例我们可以总结为几点。
① Consumer是一个接口,并且只要实现一个 accept
方法,就可以作为一个“消费者”输出信息。
② 其实,lambda 表达式、方法引用的返回值都是 Consumer 类型,所以,他们能够作为 forEach
方法的参数,并且输出一个值。
2、Supplier 接口
Supplier 接口是一个供给型的接口,其实,说白了就是一个容器,可以用来存储数据,然后可以供其他方法使用的这么一个接口。
//① 使用Supplier接口实现方法,只有一个get方法,无参数,返回一个值
Supplier<Integer> supplier = new Supplier<Integer>() {
@Override
public Integer get() {
//返回一个随机值
return new Random().nextInt();
}
};
System.out.println(supplier.get());
System.out.println("********************");
//② 使用lambda表达式,
supplier = () -> new Random().nextInt();
System.out.println(supplier.get());
System.out.println("********************");
//③ 使用方法引用
Supplier<Double> supplier2 = Math::random;
System.out.println(supplier2.get());
分析:
① Supplier接口分析
Supplier<Integer> supplier = new Supplier<Integer>() {
@Override
public Integer get() {
//返回一个随机值
return new Random().nextInt();
}
};
看一下这段代码,我们通过创建一个 Supplier 对象,实现了一个 get
方法,这个方法无参数,返回一个值;所以,每次使用这个接口的时候都会返回一个值,并且保存在这个接口中,所以说是一个容器。
② lambda表达式作为 Supplier
//② 使用lambda表达式,
supplier = () -> new Random().nextInt();
System.out.println(supplier.get());
System.out.println("********************");
上面的这段代码,我们使用 lambda 表达式返回一个 Supplier类型的接口,然后,我们调用 get
方法就可以获取这个值了。
③ 方法引用作为 Supplier
//③ 使用方法引用
Supplier<Double> supplier2 = Math::random;
System.out.println(supplier2.get());
方法引用也是返回一个Supplier类型的接口。
总结:
① Supplier 接口可以理解为一个容器,用于装数据的。
② Supplier 接口有一个 get
方法,可以返回值。
3、Predicate 接口
Predicate 接口是一个谓词型接口,其实,这个就是一个类似于 bool 类型的判断的接口。
//① 使用Predicate接口实现方法,只有一个test方法,传入一个参数,返回一个bool值
Predicate<Integer> predicate = new Predicate<Integer>() {
@Override
public boolean test(Integer integer) {
if(integer > 5){
return true;
}
return false;
}
};
System.out.println(predicate.test(6));
System.out.println("********************");
//② 使用lambda表达式,
predicate = (t) -> t > 5;
System.out.println(predicate.test(1));
System.out.println("********************");
结果:
分析:
① Predicate 接口分析
//① 使用Predicate接口实现方法,只有一个test方法,传入一个参数,返回一个bool值
Predicate<Integer> predicate = new Predicate<Integer>() {
@Override
public boolean test(Integer integer) {
if(integer > 5){
return true;
}
return false;
}
};
这段代码中,创建了一个 Predicate
接口对象,其中,实现类 test
方法,需要传入一个参数,并且返回一个 bool
值,所以这个接口作用就是判断!
System.out.println(predicate.test(6));
再看,调用 test 方法,传入一个值,就会返回一个 bool 值。
② 使用lambda表达式作为 predicate
//② 使用lambda表达式,
predicate = (t) -> t > 5;
System.out.println(predicate.test(1));
System.out.println("********************");
lambda 表达式返回一个 Predicate
接口,然后调用 test
方法!
总结:
① Predicate 是一个谓词型接口,其实只是起到一个判断作用。
② Predicate 通过实现一个 test
方法做判断。
4、Function 接口
Function 接口是一个功能型接口,它的一个作用就是转换作用,将输入数据转换成另一种形式的输出数据。
//① 使用map方法,泛型的第一个参数是转换前的类型,第二个是转化后的类型
Function<String, Integer> function = new Function<String, Integer>() {
@Override
public Integer apply(String s) {
return s.length();//获取每个字符串的长度,并且返回
}
};
Stream<String> stream = Stream.of("aaa", "bbbbb", "ccccccv");
Stream<Integer> stream1 = stream.map(function);
stream1.forEach(System.out::println);
System.out.println("********************");
效果:
分析:
① Function 接口分析
//① 使用map方法,泛型的第一个参数是转换前的类型,第二个是转化后的类型
Function<String, Integer> function = new Function<String, Integer>() {
@Override
public Integer apply(String s) {
return s.length();//获取每个字符串的长度,并且返回
}
};
这段代码创建了一个 Function
接口对象,实现了一个 apply
方法,这个方法有一个输入参数和一个输出参数。其中,泛型的第一个参数是转换前的类型,第二个是转化后的类型。
在上面的代码中,就是获取字符串的长度,然后将每个字符串的长度作为返回值返回。
② 重要应用 map 方法
Stream<String> stream = Stream.of("aaa", "bbbbb", "ccccccv");
Stream<Integer> stream1 = stream.map(function);
stream1.forEach(System.out::println);
在 Function
接口的重要应用不得不说 Stream
类的 map
方法了,map
方法传入一个 Function
接口,返回一个转换后的 Stream
类。
总结:
① Function 接口是一个功能型接口,是一个转换数据的作用。
② Function 接口实现 apply
方法来做转换。
总结
通过前面的介绍,已经对Consumer、Supplier、Predicate、Function
这几个接口有详细的了解了,其实,这几个接口并不是很难,只是有点抽象,多加理解会发现很简单,并且特别好用!