一、简述
Java8之前创建一个线程的代码:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Test");
}
});
其中Runnable就是一个匿名内部类,一般在一个线程只出现一次但必须实现。
再看Runnable的代码实现,就是一个Functional Interfaces函数式接口:
@FunctionalInterface
public interface Runnable {
public abstract void run();
}
在上面的代码中只定义了一个抽象方法,这样的接口,被称为函数式接口Functional Interface;JDK里有很多,如Predicate,Comparator。
Lambda表达式主要就是用于定义一个函数式接口的内联实现,如 Runnable r = ()-> System.out.println("Test"); ,我们用Lambda表达式实现了Runnable接口。
所以创建线程可以简写成 new Thread(()->System.out.println("Test")) 。
至于方法引用,他只是一种更简单的lambda表达式,提供了一种引用而不执方法的方式。
String::valueOf 等价于lambda表达式 (s) -> String.valueOf(s)
Lambda,Functional Interface,MethodReference,也有很多规则和约束。
二、规则
1、Lambda
Lambda表达式具有如下特征:
- 【可选】类型声明:参数的类型不需要声明,编译器可以根据参数值推断出其类型;
- 【可选】括号:单个参数的话,不需要用圆括号包围参数,当然,对于多个参数或无参数的话,括号是需要的;
- 【可选】花括号:如果表达式主体只有一条语句的话,不需要用花括号包围,当然,对于多条语句,花括号是需要的;
- 【可选】return关键字:如果表达式主体是单一表达式,return关键字可以不写,编译器可以自动返回该值,当然,如果写了return,则需要加上花括号;
2、函数式接口
- 只能有一个抽象方法。
- @FunctionalInterface,主要用于编译级错误检查,当你写的接口不符合函数式接口定义的时候,编译器会报错,加不加@FunctionalInterface对于接口是不是函数式接口没有影响
- 可以包含默认方法,因为默认方法不是抽象方法,其有一个默认实现,所以是符合函数式接口的定义的;default关键字
- 可以包含静态方法,因为静态方法不能是抽象方法,是一个已经实现了的方法,所以是符合函数式接口的定义的;
- 可以包含Object里的public方法,这些方法对于函数式接口来说,不被当成是抽象方法(虽然它们是抽象方法);因为任何一个函数式接口的实现,默认都继承了Object类,包含了来自java.lang.Object里对这些抽象方法的实现;
3、方法引用
四种方法引用类型
类型
示例
引用静态方法
ContainingClass::staticMethodName
引用某个对象的实例方法
containingObject::instanceMethodName
引用某个类型的任意对象的实例方法
ContainingType::methodName
引用构造方法
ClassName::new
a、静态方法引用
就是静态方法可以用
b、特定对象实例方法
Student s = new Student()
Arrays.sort(students,s::getAge)super::methodNamethis :: equals
c、任意对象的实例方法引用
String[] stringArray = { "Barbara", "James", "Mary", "John", "Patricia", "Robert", "Michael", "Linda" };
Arrays.sort(stringArray, String::compareToIgnoreCase);
d、构造方法引用
String::new, 等价于lambda表达式 () -> new String()
int[]::new 是一个含有一个参数的构造器引用,这个参数就是数组的长度。等价于lambda表达式 x -> new int[x]