摘要:表达式又名闭包匿名函数笔记根据终于在中引入了表达式。函数式接口要介绍中表达式的实现,需要知道什么是函数式接口。但同样需要保证外部的自由变量不能在表达式中被改变。
Java Lambda 表达式(又名闭包 (Closure)/ 匿名函数 ) 笔记
根据 JSR 335, Java 终于在 Java 8 中引入了 Lambda 表达式。也称之为闭包或者匿名函数。
JSR 335所谓的 JSR (Java Specification Requests) 全称叫做 Java 规范提案。简单来说就是向 Java 社区提交新的 API 或 服务 请求的提案。这些提案将作为 Java 社区进行 Java 语言开发的需求,引导着开发的方向。
JSR 335 的提案内容摘要如下:
This JSR will extend the Java Programming Language Specification and the Java Virtual Machine Specification to support the following features:
ambda Expressions
SAM Conversion
Method References
Virtual Extension Methods
也就是如下几点:
支持 lambda 表达式。
支持 SAM conversion 用来向前兼容。
方法引用 Method References
Virtual Extension Methods
在 Java 8 中,以上均已经实现, 以上内容下文均有介绍。
为什么需要 Lambda 表达式?Lambda 表达式,其实就是代码块。
原来怎么处理在具体了解 lambda 之前,我们先往后退一步,看看之前我们是如何处理这些代码块的!
例子一当决定在多带带的线程运行某程序时,你这样做的
class Worker implements Runnable { public void run() { for (int i = 0; i < 1000; i++) doWork(); } ... }
这样执行:
Worker w = new Worker(); new Thread(w).start();
Worker 中包含了你要执行的代码块。
例子二如果你想实现根据字符串长度大小来排序,而不是默认的字母顺序,你可以自己来实现一个 Comparator 用来 Sort
class LengthComparator implements Comparator{ public int compare(String first, String second) { return Integer.compare(first.length(), second.length()); } } Arrays.sort(strings, new LengthComparator());
####例子三
另外一个例子,我选的是 Android 中的点击事件,同样是 Java:
button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { Toast.makeText(MainActivity.this, "Hello World!", Toast.LENGTH_SHORT).show(); } });
###上面代码有什么问题呢?###
它们都太复杂了啊!
上述例子都是在某个类中实现某个接口,然后传递到另外一个方法中作为参数,然后用来执行。
但是本质上,他们要传递的就是接口中那一个方法的实现而已啊!有必要先创建类,再实例化,再传递给调用的位置吗?
因为 Java 是纯面向对象的语言,像其他语言那样随随便便传个方法过来,那可不行,必须要这样。
在其他语言中你可能可以,但是,在Java 中,不可以。
Java 设计人员为了 Java 的简洁跟连贯性,一直拒绝为 Java 添加这种功能。(这也是我喜欢 Java 而不喜欢 Python 的原因啊!!!)
经过多年的努力,开发人员终于找到了符合 Java 编程习惯的 Lambda 表达式!
Lambda 表达式语法(Syntax)考虑下前面的例子:
Integer.compare(first.length(), second.length())
first 和 second 都是 String 类型,Java 是强类型的语言,必须指定类型:
(String first, String second) -> Integer.compare(first.length(), second.length())
看到没有!第一个 Lambda 表达式诞生了!!输入、输出简洁明了!
为什么叫 Lambda 呢,这个很多年以前,有位逻辑学家想要标准化的表示一些可以被计算的数学方程(实际上存在,但是很难被表示出来),他就用 ℷ 来表示。
重新介绍一下 Java 中 Lambda 表达式的格式:
多返回值(参数) -> 表达式
(String first, String second) -> { if (first.length() < second.length()) return -1; else if (first.length() > second.length()) return 1; else return 0;
}
如果计算的结果并不由一个单一的表达式返回(换言之,返回值存在多种情况),使用“{}”,然后明确指定返回值。
(String first, String second) -> { if (first.length() < second.length()) return -1; else if (first.length() > second.length()) return 1; else return 0; }
###无参数
如果没有参数,则 “()”中就空着。
() -> { for (int i = 0; i < 1000; i++) doWork(); }
###省略###
Comparatorcomp = (first, second) // Same as (String first, String second) -> Integer.compare(first.length(), second.length());
这里,first 和 second 可以被推断出是 String 类型,因为 是一个 String 类型的 Comparator。
如果单个参数可以被推断出,你连括号都可以省略:
EventHandlerlistener = event -> System.out.println("Thanks for clicking!"); // Instead of (event) -> or (ActionEvent event) ->
###修饰符
你可以像对待其他方法一样,annotation,或者 使用 final 修饰符
(final String name) -> ... (@NonNull String name) -> ...
永远不要 定义 result 的类型,lambda 表达式总是从上下文中推断出来的:
(String first, String second) -> Integer.compare(first.length(), second.length())
###注意
注意,在 lambda 表达式中,某些分支存在返回值,某些不存在返回值这样的情况是不允许的。
如 (int x) -> {if (x >= 0) return 1; }这样是非法的。
要介绍 Java 中 lambda 表达式的实现,需要知道什么是 函数式接口。
什么叫作函数式接口呢 (SAM)?
函数式接口指的是只定义了唯一的抽象方法的接口(除了隐含的 Object 对象的公共方法), 因此最开始也就做 SAM 类型的接口(Single Abstract Method)。
Lambda 表达式 向前兼容 这些接口。
Comparable举个例子 Array.sort:
Arrays.sort(words, (first, second) -> Integer.compare(first.length(), second.length()));
Array.sort() 方法收到一个实现了 Comparable接口的实例。
其实可以把 Lambda 表达式想象成一个方法,而非一个对象,一个可以传入一个接口的方法。
OnClickListener再举个例子
button.setOnClickListener(event -> System.out.println("Thanks for clicking!"));
你看,是不是更易读了呢?
Lambda 表达式能够向前兼容这些 interfaces, 太棒了! 那 Lambda 表达式还能干什么呢?
实际上,将函数式接口转变成 lambda 表达式是你在 Java 中 唯一 能做的事情。
Why ?!!
在其他的语言中,你可以定义一些方便的方法类型,但在 Java 中,你甚至不能将一个 Lambda 表达式赋值给类型为 Object 的变量,因为 Object 变量不是一个 Functional Interface。
Java 的设计者们坚持使用熟悉的 interface 概念而不是为其引入新的 方法类型。
(这里我还要为设计者点赞!谨慎的设计,一方面降低了初学者的门槛,一方面方便了高级用户的使用。对比 python2 和 python3,升级的不兼容让很多人一直停留在 python2)
Method References能不能再简洁一点?有的时候我们所要做的事情不过是调用其他类中方法来处理事件。
button.setOnClickListener(event -> System.out.println(event));
如果这样呢?
button.setOnAction(System.out::println);
表达式 System.out::println 属于一个方法引用(method reference), 相当于 lambda 表达式 x -> System.out.println(x)
再举个例子,如果你想对字符串不管大小写进行排序, 就可以这样写!
Arrays.sort(strings, String::compareToIgnoreCase)
如上所见 ::操作符将方法名与实例或者类分隔开。总体来说,又如下的规则:
object::instanceMethod
Class::staticMethod
Class::instanceMethod
值得指出的是, this和 super 关键字可以在其中使用:
class Greeter { public void greet() { System.out.println("Hello, world!"); } }
.
class ConcurrentGreeter extends Greeter { public void greet() { Thread t = new Thread(super::greet); t.start(); } }构造方法引用 Constructor References
跟上一个差不多,毕竟 构造方法 也是方法啊!!不过方法名字为 new 。
但是!这个构造方法引用有一个牛逼的地方!
你知道 Array 是不能使用范型的对吧!(什么,你不知道?看看这里 http://stackoverflow.com/ques... 你没有办法创建一个类型为 T 的 Array 。 new T[n] 将会被覆盖为 new Object[n]。
假设我们想要一个包含 buttons 的 Array。Stream interface 可以返回一个 Object array。
Object[] buttons = stream.toArray();
不不不,我们可不想要 Object。Stream 库使用 构造方法引用解决了这个问题:
Button[] buttons = stream.toArray(Button[]::new);变量作用域
注意到我们在题目中写着 闭包(closure), 实际上,闭包的定义是: 引用了自由变量的函数。
在之前,如果需要在匿名类的内部引用外部变量,需要将外部变量定义为 final ,现在有了 lambda 表达式,你不必再这么做了。但同样需要保证外部的自由变量不能在 lambda 表达式中被改变。
这是什么意思呢? 不需要定义为 final,也不能改?
其实理解起来很简单,Java 8 中,不需要定义为 final ,但你其实可以直接把他当作 final,不要试图修改它就行了。
即便你用内部类,现在也无需定义为 final 了。
参考 StackOverFlow 链接: http://stackoverflow.com/ques...
Default Methods由于历史原因,像是类似 Collection 这种接口,如果进行添加接口的话,那将会造成之前的代码出错。
Java 想了一个一劳永逸的方法解决这个问题, 使用 default 修饰符来提供默认的实现
比如 Collection 接口的源代码:
default void remove() { throw new UnsupportedOperationException("remove"); }
当没有 override remove 这个方法是,调用的时候返回 UnsupportedOperationException 错误。
Static Methods in InterfacesJava 8 中,你可以在接口中添加静态方法了。 看起来好像并不符合接口的定义了。
一般用来生成一个简单实现该 interface 的实例。
参考链接:
JSR 335: Lambda Expressions for the JavaTM Programming Language
Java 8 新特性概述
Lambda Expressions in Java 8
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/67955.html
摘要:恩如期来啦闭包一函数作为返回值介绍闭包之前,先了解一下函数作为返回值的情况。例如之前介绍的装饰器中,就出现了将函数作为返回值。当执行时,相当于执行,且包含。允许使用关键字创造匿名函数。例如调用默认可以把匿名函数作为返回值返回,例如 恩~ 如期来啦闭包~ 一、函数作为返回值 介绍闭包之前,先了解一下函数作为返回值的情况。高阶函数除了可以接收函数作为参数外,还可以把函数作为结果值返回。...
摘要:进阶细节根据慕课网七月老师视频整理一切皆对象对与来说,一切皆对象,包括函数。闭包的意义在于返回了一个现场,如果没有闭包,很容易被外部的变量所影响。重复定义可见此时返回了不再是闭包了。注意该方法的返回值只能是布尔类型,即或。 Python进阶细节 根据慕课网七月老师视频整理 一切皆对象 对与Python来说,一切皆对象,包括函数。在其他语言比如c++中,函数只是一段可执行的代码,只要你获...
摘要:之前,使用匿名类给苹果排序的代码是的,这段代码看上去并不是那么的清晰明了,使用表达式改进后或者是不得不承认,代码看起来跟清晰了。这是由泛型接口内部实现方式造成的。 # Lambda表达式在《Java8实战》中第三章主要讲的是Lambda表达式,在上一章节的笔记中我们利用了行为参数化来因对不断变化的需求,最后我们也使用到了Lambda,通过表达式为我们简化了很多代码从而极大地提高了我们的...
摘要:的语法和其它流程控制结构相似部分允许设定代码段的行为。返回值在失败时返回并且发出警告。当一个函数是有条件被定义时,必须在调用函数之前定义。有条件的函数不能在此处调用函数,因为它还不存在,但可以调用函数。 流程控制 PHP 提供了一些流程控制的替代语法,包括 if,while,for,foreach 和 switch。替代语法的基本形式是把左花括号({)换成冒号(:),把右花括号(})分...
阅读 2896·2021-11-11 16:55
阅读 943·2021-09-28 09:36
阅读 3792·2021-09-22 15:22
阅读 2222·2021-09-06 15:12
阅读 1753·2021-08-19 10:55
阅读 2887·2019-08-30 12:52
阅读 494·2019-08-29 14:03
阅读 1203·2019-08-29 12:27