Java Lambda 表达式介绍

Posted Heaven-Wang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java Lambda 表达式介绍相关的知识,希望对你有一定的参考价值。

Lambda 表达式是 Java SE8 推出的新功能,也是Java第一次引入函数式编程的尝试。

Lambda表达式格式

Lambda 表达式可以看做是一种匿名函数,但是它没有访问修饰符、返回值和名字。Lambda表达式由两部分构成,形式参数和方法体,中间用“->”符号分隔。其中的形式参数类型能够进行自动推断,可以不写。当然在某些特殊情况下,形参类型也是不可缺少的。方法体可以是简单的表达式或者代码块,下面是一些例子:

// 1. 不需要参数,返回值为 5
() -> 5
// 2. 接收一个参数(数字类型),返回其2倍的值
x -> 2 * x
// 3. 接受2个参数(数字),并返回他们的差值
(x, y) -> x – y
// 4. 接收2个int型整数,返回他们的和
(int x, int y) -> x + y
// 5. 接受一个 string 对象,并在控制台打印,不返回任何值(看起来像是返回void)
(String s) -> System.out.print(s)

函数式接口

函数式接口

要理解Lambda表达式,首先要了解一种特殊的接口:函数式接口。什么是函数式接口呢?简单来说就是只包含一个抽象方法的接口。Java标准库中的java.lang.Runnablejava.util.Comparator就是典型的函数式接口。对于函数式接口,我们就可以使用Lambda表达式来代替用传统匿名类来创建实例对象。

我们以Runnable接口为例,用传统匿名类的方式创建一个线程:

public void runThread() 
    new Thread(new Runnable() 
        public void run() 
            System.out.println("Run!");
        
    ).start();

上面传统匿名类方式中,我们可以看到,我们需要new一个接口名称,接口内部还要附带这个接口抽象方法的实现。而如果我们使用Lambda表达式,则代码非常简洁:

public void runThreadUseLambda() 
    new Thread(() -> 
        System.out.println("Run!");
    ).start();

通过上面代码我们可以看到,Lambda表达式在两方面做了简化:

  • 首先不需要声明Runnable接口,因为这可以通过上下文推断出来
  • 其次不需要再写一个run方法的实现,因为函数式接口中只有一个方法

java.util.function包

在Java SE8之前标准库中的函数式接口并不多。JavaSE8增加了java.util.function包,里面都是可以在开发中只用的函数式接口。我们也可以自定义一个函数式接口,但最好在接口上使用@FunctionalInterface注解标明这是一个函数式接口,以免团队其它成员错误地往接口添加新的方法,当然java.util.function包中的所有接口都添加了@FunctionalInterface注解。

下面代码使用函数式接口java.util.Function接口实现的对列表map进行操作的方法,从代码中可以看出,如果使用函数式编程,代码看起来会非常简洁:

public class CollectionUtils 
    public static  List map(List input, Function processor) 
        ArrayList result = new ArrayList();
        for (T obj : input) 
            result.add(processor.apply(obj));
        
        return result;
    

    public static void main(String[] args) 
        List input = Arrays.asList(new String[] "apple", "orange", "pear");
        List lengths = CollectionUtils.map(input, (String v) -> v.length());
        List uppercases = CollectionUtils.map(input, (String v) -> v.toUpperCase());
    

自定义函数式接口及@FunctionalInterface注解

注解可以起到指示、约束作用,关于注解大家可以参考 Java 注解详解 (annotation)。Java 8 为函数式接口新提供了@FunctionalInterface注解,当用注解的接口不是有效的函数式接口时,也就是接口中不只有一个抽象方法时,编译器会报编译错误。我们自定义一个函数式接口,并用@FunctionalInterface进行注解

@FunctionalInterface
public interface MyFunction 
    public void doSomething();

如果我们在上面接口中再加一个抽象方法:

@FunctionalInterface
public interface MyFunction 
    public void doSomething();
    public void doAnotherThing();

由于自定义接口中出现了两个抽象方法,不符合函数式接口定义,此时会抛出编译错误。我们再测一下上面这个自定义函数式接口:

public class TestMyFunction 
    public static void execute(MyFunction worker) 
        worker.doSomething();
    
    public static void main(String [] args) 
        //传统匿名类的方式调用
        execute(new MyFunction() 
            public void doSomething() 
                System.out.println("Worker invoked using Anonymous class");
            
        );
        //通过Lambda表达式调用
    execute( () -> System.out.println("Worker invoked using Lambda expression") );
    

以上是关于Java Lambda 表达式介绍的主要内容,如果未能解决你的问题,请参考以下文章

Kotlin函数 ⑤ ( 匿名函数变量类型推断 | 匿名函数参数类型自动推断 | 匿名函数又称为 Lambda 表达式 )

Java Lambda表达式入门

Java中Lambda表达式的使用

Java中Lambda表达式的使用

Java中Lambda表达式的使用

Java8新特性之lambda表达式