Java 8中的Lambda表达式


Lambda表达式是Java 8的最流行功能。它们将函数式编程概念引入Java,这是一种完全面向对象的命令式编程语言。函数式编程语言的工作方式不在本文讨论范围之内,但是我们将提取一项功能,这对于使用OOP的我们来说将是显而易见的。

函数式编程语言的工作方式不在本文讨论范围之内,但是我们将提取一项功能,这对于使用OOP的我们来说将是显而易见的。

在本文中,我们将确切地了解lambda表达式是什么以及它们如何适合整个Java生态系统。我们还将查看示例代码,这些代码无需使用lambda表达式即可工作,然后将其重构为使用lambda。

了解Lambda表达式

Lambda表达式是我们可以传递来执行的代码块。作为Java程序员,我们不习惯将代码块传递给函数。我们所有的行为定义代码都封装在方法体内,并通过对象引用执行,就像使用此代码一样:

public class LambdaDemo {
    public void printSomething(String something) {
        System.out.println(something);
    }

    public static void main(String[] args) {
        LambdaDemo demo = new LambdaDemo();
        String something = "I am learning Lambda";
        demo.printSomething(something);
    }
}

这是对调用者隐藏方法实现的经典OOP样式。调用者只需将一个变量传递给该方法,该方法然后对该变量的值进行处理,然后返回另一个值或产生副作用(如本例所示)。

现在,我们将看到使用行为传递而不是变量传递的等效实现。为此,我们必须创建一个定义抽象功能而不是方法的功能接口。功能接口是只有一种方法的接口:

public class LambdaDemo {
    interface Printer {
        void print(String val);
    }

    public void printSomething(String something, Printer printer) {
        printer.print(something);
    }
}

在上述实现中,打印机接口负责所有打印。方法printSomething不再定义行为,而是执行Printer定义的行为:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am using a Functional interface";
    Printer printer = new Printer() {
        @Override
        public void print(String val) {
            System.out.println(val);
        }
    };
    demo.printSomething(something, printer);
}

你们中间的观察者可能已经注意到我们在这里没有做新的事情。这是正确的,因为我们尚未应用lambda表达式。我们仅创建了Printer接口的具体实现,并将其传递给printSomething方法。

上面的演示旨在使我们达到在Java中引入Lambda表达式的关键目标:

Lambda表达式主要用于定义功能接口的内联实现。

在使用lambda表达式重构上述示例之前,让我们学习必要的语法:

(param1,param2,param3...,paramN) - > {//block of code;}

一个lambda构成一个用逗号分隔的形式参数列表,该形式参数括在括号中,正如我们在方法声明中所定义的那样,后跟一个箭头标记,指向要执行的代码。现在,让我们重构上面的代码以使用lambda:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am learning Lambda";
    /**/
    Printer printer = (String toPrint)->{System.out.println(toPrint);};
    /**/
    demo.printSomething(something, printer);
}

非常紧凑,外观漂亮。由于功能接口仅声明单个方法,因此在lambda的第一部分中传递的参数将自动映射到该方法的参数列表,并且箭头右侧的代码被视为该方法的具体实现。

为什么使用Lambda表达式 与上一节中的演示有关,lambda表达式使我们能够拥有更紧凑的代码,更易于阅读和遵循。性能和多核处理还有其他好处,但只有在理解Streams API之后才有意义,因此不在本文讨论范围之内。

比较带有和不带有lambda的主要方法实现,可以肯定地向我们展示了lambda表达式在缩短代码方面的强大功能:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am learning Lambda";
    /**/
    Printer printer = (String toPrint)->{System.out.println(toPrint);};
    /**/
    demo.printSomething(something, printer);
}

我们可以使我们的代码比这更简洁。碰巧的是,即使没有在箭头的左侧指定参数的类型,编译器也会从接口方法的形式参数中推断出其类型:

Printer printer = (toPrint)->{System.out.println(toPrint);};

我们仍然可以做得更好。lambda的另一个特征是:如果只有一个参数,则可以完全消除括号。同样,如果箭头的右边只有一个语句,我们也可以消除花括号:

Printer printer = toPrint -> System.out.println(toPrint);

现在我们的代码真的开始看起来很可爱了。我们才刚刚开始。如果我们的接口方法不带任何参数,则可以用空括号替换声明:

() -> System.out.println("anything");

不用先创建一个对象然后将其传递给saySomething方法,就可以内联lambda怎么样:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something="I am Lambda";
    /**/
    demo.printSomething(something, toPrint -> System.out.println(toPrint));
}

现在我们真的开始谈论函数式编程了。我们最初主要的九行体是现在下降到只有3行。代码的这种紧凑性使lambda表达式非常吸引Java程序员。

结论 在本文中,我们对Java中的Lambda表达式进行了详尽的介绍,并了解了如何使用它们来提高功能接口实现的代码质量。请注意本网站上有关Lambda的更多报道,因为我将接触Stream API,并讨论将其与Collections框架一起使用时如何为Lambda提供更多优势。


原文链接:http://codingdict.com