Java8 Lambda( -> )&&方法引用( :: )&&函数式接口(@Functional)

一、简述

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表达式具有如下特征:

扫描二维码关注公众号,回复: 3900683 查看本文章
  • 【可选】类型声明:参数的类型不需要声明,编译器可以根据参数值推断出其类型;
  • 【可选】括号:单个参数的话,不需要用圆括号包围参数,当然,对于多个参数或无参数的话,括号是需要的;
  • 【可选】花括号:如果表达式主体只有一条语句的话,不需要用花括号包围,当然,对于多条语句,花括号是需要的;
  • 【可选】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::methodName
this :: 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]

猜你喜欢

转载自www.cnblogs.com/Samuel1/p/9897658.html