在这篇文章中,我们将了解到在Java8下如何进行函数式编程。
函数式编程
所谓的函数式编程就是把函数名字当做值进行传递,然后接收方拿到这个函数名进行调用。
首先来看下JavaScript如何进行函数调用
var Person = {
sayHello: function(name) {
alert('hello ' + name);
}
};
function work(fn) {
fn('Jim');
}
work(Person.sayHello); // hello Jim
在这个例子中,sayHello
函数被当做一个参数传递到另一个work
函数当中,然后work调用给定的函数。
Java8中的函数式编程
在Java中,充当函数的角色是类中方法,在本篇文章当中提到的函数泛指方法。
接下来看下Java8中一个简单的函数式编程例子:
import java.util.function.Consumer;
class Person {
public static void sayHello(String name) {
System.out.println("hello " + name);
}
}
public class TestPerson {
public static void main(String[] args) {
work(Person::sayHello); // hello Jim
}
public static void work(Consumer<String> consumer) {
consumer.accept("Jim");
}
}
从这个例子中可以看到,Java8传递函数的方式跟JavaScript不一样,在Java8当中传递函数的方式是:方法引用::方法名称
,如:String.valueOf,String.toString。接收参数Consumer可以先不用去了解,后面会讲到。
为了接收传递过来的函数引用,Java设计者提出了一个函数式接口
的概念
函数式接口
函数式接口定义:一个接口里面只有一个抽象方法
。
如下面几个接口都是函数式接口:
- Consumer
- Supplier
- Runnable
- Callable
查看Runnable类的源码可以发现,在类上方定义了一个@FunctionalInterface
注解
@FunctionalInterface
public interface Runnable {
public abstract void run();
}
- @FunctionalInterface注解的作用:
当做一个标记使用,标记当前接口是一个函数式接口,如果你的接口有多个抽象方法或没有抽象方法,那么会编译报错。它的作用有点类似于@override
注解,有辅助作用,但不是必须的。如果你想设计一个函数式接口,那么最好把它加上。
// 编译不通过,必须要有一个抽象方法
@FunctionalInterface
public interface InterfaceA {
}
// 编译通过,只有一个抽象方法
@FunctionalInterface
public interface InterfaceB {
void run();
}
// 编译不通过,只能有一个抽象方法
@FunctionalInterface
public interface InterfaceC {
void run();
void run2();
}
// 编译通过,run2是默认方法,并不是抽象方法
// 在这里只有一个run抽象方法
@FunctionalInterface
public interface InterfaceD {
void run();
default void run2() {};
}
在例子InterfaceD中涉及到了接口默认方法default void run2() {};
,这里略过不讲,后续文章中会有提到。
我们可以实现一个自定义的函数式接口,替代Consumer
// 自定义函数式接口
@FunctionalInterface
public interface FunctionCaller<T> {
void call(T t);
}
public class TestPerson2 {
public static void main(String[] args) {
work(Person::sayHello);
}
public static void work(FunctionCaller<String> caller) {
caller.call("Jim");
}
}
小结
本篇主要讲解了在Java8下如何进行函数式编程,以及如何定义和使用一个函数式接口。
定期分享技术干货,一起学习,一起进步!