Java 8 中的 Lambda 表达式 已翻译 100%

oschina 投递于 2016/12/06 10:57 (共 6 段, 翻译完成于 12-06)
阅读 3854
收藏 57
9
加载中

Lambda 表达式是 Java 8 最受欢迎的功能。人们将函数式编程的概念引入了 Java 这门完全面向对象的命令式编程语言。关于函数式编程是如何运作的,这个话题超出了本文的范围,不过我们会提炼出它一个明显有别于我们所经常使用的 OOP (面向对象编程)的功能来加以讨论。

在本文中, 我们将了解到 lambda 表达式具体是什么东西,还有就是它们是如何将自己融入整个 Java 生态系统的。我们也会对没有使用 lambda 表达式的代码以及后面使用 lambda 进行重构的示例代码进行一下观察和比较。

LeoXu
翻译于 2016/12/06 11:18
0

了解 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 开发范式的风格,将方法实现对调用者隐藏。调用者只是简单地向方法传入一个变量,然后方法拿这个变量会执行一些操作,并返回另外一个变量值,或者如我们的示例所示,会产生一些副作用效果。

LeoXu
翻译于 2016/12/06 11:26
0

现在我们要来看看一种使用了行为传递方式,而不是变量传递的等效实现。为此,我们要创建一个函数式的接口,里面定义的是对行为,而不是对方法的抽象。一个函数式接口是一种只有一个方法的接口:

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

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

在上面的代码实现中, Printer 接口负责所有的打印操作。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 方法。

LeoXu
翻译于 2016/12/06 11:34
0

上面的示例旨在给我们带来一个将 Lambda 表达式引入到 Java 中的关键目标:

Lambda 表达式原被用于定义一个函数式接口的内联实现。

在我们使用 lambda 表达式对上面的示例进行重构之前,先来学习一下必要的语法知识:

(param1,param2,param3...,paramN) - > {//代码块;}

一个 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 的第一部分中传入的参数被自动地映射到了方法的参数列表上,而箭头右边的代码则被当做是方法的具体实现了。

LeoXu
翻译于 2016/12/06 11:46
0

为什么要使用 Lambda 表达式

如同前面的示例, lambda 表达式能让我们拥有更加紧凑的代码,更加易于阅读和跟踪。这个在性能和多核处理方法还有其它的一些好处,不过它们得在你了解了 Streams API 以后才有用,而这个超出了本文的范围。

通过比较使用和没使用 lambda 的 main 方式实现,当它一下子把代码变得简短的时候,我们切实地看到了 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");

如果我们只是内联一个 lambda 进去,而不去首先创建一个对象然后将其传入到 saySomething 方法,会如何呢:

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

现在我们才是真的在谈论函数式编程了。我们的 main 函数体从一开始的 9 行代码减少到了 3 行。这样紧凑的代码使得 lambda 表达式对于 Java 程序员非常有吸引力。

LeoXu
翻译于 2016/12/06 12:01
1

总结

在本文中,我们对 Java 中的 Lambda 表达式进行了简单介绍,了解了它们可以被用来提升函数式接口实现的代码质量。关注这个网站可以获得有关 Lambda 的更多知识,因为我还会在这里涉及 Stream API 的内容,并对其如何同 Collections 框架结合在一起使用来给予我们更多 Lambda 的好处进行讨论。

LeoXu
翻译于 2016/12/06 12:06
1
本文中的所有译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接。
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。
加载中

评论(28)

布客飞龙
布客飞龙
Python:f = lambda i: i ** 2
JavaScript:var f = function(i) { return i * i; };
C#:Func<int, int> f = delegate(int i) { return i * i; };

Java8:汪汪汪?
MikeManilone
MikeManilone

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));
PHP在5.3才支持匿名函数呀,LISP在1958年就支持了。
ramnight
ramnight

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。

引用来自“eechen”的评论

来来来,我很想看看Ruby是如何秒杀PHP的. Ruby在PHP面前只剩下装逼,比Java还可怜.

引用来自“ramnight”的评论

做web用php,玩语法用ruby,看前景就学go,做大项目还得是java。

引用来自“eechen”的评论

JAVAer就别迷之自信了,Java小项目做起来都那么麻烦还怎么做大项目? 一屋都扫不好,还妄想横扫天下,笑死人. 还有Ruby语法好?呵呵,你干嘛不去看看Hyperpolyglot列出的PHP/JS/Python/Ruby对比呢? Golang有前景?在哪?连Golang官方都说,学Golang最多的是那些Python用户,而不是C/C++.所以呢,Golang在Web上干不过PHP/JS,底层干不过C/C++,有前景个蛋呀,以为跟着Google就有前景呀,Google自己的Dart都半死不活了.
屌,php秒世界,满意了吧?
eechen
eechen

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。

引用来自“eechen”的评论

来来来,我很想看看Ruby是如何秒杀PHP的. Ruby在PHP面前只剩下装逼,比Java还可怜.

引用来自“ramnight”的评论

做web用php,玩语法用ruby,看前景就学go,做大项目还得是java。
JAVAer就别迷之自信了,Java小项目做起来都那么麻烦还怎么做大项目? 一屋都扫不好,还妄想横扫天下,笑死人. 还有Ruby语法好?呵呵,你干嘛不去看看Hyperpolyglot列出的PHP/JS/Python/Ruby对比呢? Golang有前景?在哪?连Golang官方都说,学Golang最多的是那些Python用户,而不是C/C++.所以呢,Golang在Web上干不过PHP/JS,底层干不过C/C++,有前景个蛋呀,以为跟着Google就有前景呀,Google自己的Dart都半死不活了.
java干货
java干货
又稍微了解一下
ramnight
ramnight

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。

引用来自“eechen”的评论

来来来,我很想看看Ruby是如何秒杀PHP的. Ruby在PHP面前只剩下装逼,比Java还可怜.
做web用php,玩语法用ruby,看前景就学go,做大项目还得是java。
看星空的老蚂蚁
看星空的老蚂蚁

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。

引用来自“jasonwu24”的评论

总有些拍黄片程序员在Java世界里拍来拍去。。。
上面这段代码还算不上语法糖吧, 只要指持函数指针的都能这么干
eechen
eechen

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。
来来来,我很想看看Ruby是如何秒杀PHP的. Ruby在PHP面前只剩下装逼,比Java还可怜.
jasonwu24
jasonwu24

引用来自“eechen”的评论

J8才支持匿名函数呀,真落伍.
要知道,PHP从5.3开始就支持了.
$func = array(
  'foo' => function($arg){echo $arg;},
  'bar' => function($arg){echo $arg;}
);
call_user_func_array($func['foo'], array('php'));
call_user_func_array($func['bar'], array('best'));

引用来自“ramnight”的评论

碉堡啊,java的优势不是语法糖,要比语法糖,有本事拿你的php去跟ruby去比啊,秒杀php一万条街。
总有些拍黄片程序员在Java世界里拍来拍去。。。
SungineOS
SungineOS
非复用的场景,lamda挺给力的
返回顶部
顶部