Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

目录

Lambda 表达式

Lambda 表达式也称为闭包,是匿名内部类的简短形式。

Lambda 表达式简化了单一方法声明接口的使用,因此 lambda 表达式也称为功能接口(只有一个抽象方法的接口)。

使用 lambda 表达式实现功能接口时,无需创建类或匿名类。

Lambda 表达式具有以下优点:

  • 简明的语法
  • 方法引用和构造函数引用
  • 相比于匿名类,减少了运行时开销

基本语法

(formal parameter list) -> {
expression or statements
}

参数列表

  • 参数列表是一个逗号分隔的形式参数列表,这些参数与功能接口中单一方法的形式参数相对应。
  • 参数列表中的参数类型是可选项,如果未指定参数类型,将从上下文推断。
  • 参数列表必须用小括号括起来,但如果只有一个参数且不带参数类型时小括号可以省略
  • 参数列表如果为空(即:功能接口方法没有形式参数),则必须指定空括号

Lambda 主体根据以下选项之一返回结果:

  • 如果 lambda 主体是单一表达式,则返回表达式的值(如果有的话)。
  • 如果功能接口方法的结果是 void,可以提供一个 return 语句,但这不是必需的。
  • 如果功能接口方法具有返回类型,且 lambda 主体不是单一表达式,则 lambda 主体必须使用 return 语句返回匹配的值。

语句块必须包含在大括号内,除非语句块是一个方法调用语句,且功能接口方法的返回结果是void。

Lambda 表达式实际上是一种匿名方法实现,指定形式参数,并使用 return 语句返回值。匿名方法必须按照以下规则所规定的与其实现的功能接口方法兼容。

  • Lambda 表达式返回的结果必须与功能接口方法的结果兼容。返回值的类型可以是功能接口方法声明中返回类型的子类型。
  • Lambda 表达式签名必须与功能接口方法的签名相同。
  • Lambda 表达式只能抛出那些在功能接口方法的 throws 子句中声明了异常类型或异常超类型的异常。

Lambda 表达式中的局部变量

Lambda 表达式不会定义新的作用域,lambda 表达式的作用域与封闭作用域相同。

Lambda 主体中的 this 和 super 引用与封闭上下文中一样,因为 lambda 表达式不会引入新的作用域,这与匿名类不同。

如果 Lambda 主体声明的局部变量与封闭作用域内的变量重名,将产生编译器错误

Lambda expression's local variable i cannot re-declare another local variable defined in an enclosing scope

局部变量无论是在 lambda 表达式主体中声明,还是在封闭作用域中声明,使用之前都必须先初始化,否则将产生编译器错误

The local variable i may not have been initialized

lambda 表达式中使用的变量必须处于终态或等效终态,否则将产生编译器错误

Variable i is required to be final or effectively final

启用 Lambda 表达式

在AS中使用 Lambda 表达式

File -> Project Structure -> SDK location -> JDK location -> JDK版本选择1.8

android {
compileOptions {
targetCompatibility JavaVersion.VERSION_1_8
sourceCompatibility JavaVersion.VERSION_1_8
}
}

在Eclipse中使用 Lambda 表达式

前提条件

  • 1、安装了 JDK 8 的 JRE
  • 2、所使用的 Eclipse 要支持 Java8 的编译

步骤

  • 在 Eclipse 中,选择 Windows > Preferences > Java > Installed JREs,勾选JDK 8 的 JRE
  • 选择 Windows > Preferences > Java > Compiler,然后将 Compiler compliance level 设为 1.8。如果没有 1.8 的选项,说明此eclipse版本不支持。
  • 单击 Apply,然后单击 OK。

Eclipse Luna SR2已经添加了对Java8的支持,可直接到官网下载

使用示例

没有参数

new Thread(() -> {
System.out.println("包青天");
}).start();

只有一行语句时,可以进一步简化为:

new Thread(() -> System.out.println("包青天")).start();
new Thread(() -> runOnUiThread(() -> System.out.println("包青天"))).start();

但是如果有多行语句,就没法简化了

单一参数

imageView.setOnClickListener(v -> {
Toast.makeText(this, "包青天", Toast.LENGTH_SHORT).show();
});

同样,只有一行语句时,可以进一步简化为:

imageView.setOnClickListener((View v) -> System.out.println("包青天")); //带类型
imageView.setOnClickListener((v) -> System.out.println("包青天")); //带小括号
imageView.setOnClickListener(v -> System.out.println("包青天")); //省略小括号
imageView.setOnClickListener(v -> System.out.println(v));

如果只有一行方法引用的语句,且此方法没有返回值,且此方法仅有一个参数,且此方法的此参数就是 Lambda 表达式中参数列表中的那个参数,可以进一步简化为方法引用(method reference)

imageView.setOnClickListener(v -> test(v));

imageView.setOnClickListener(this::test); //普通方法
imageView.setOnClickListener(System.out::println); //静态方法
imageView.setOnClickListener(SplashActivity::test); //静态方法

多个参数

listView.setOnItemClickListener((AdapterView<?> parent, View view, int position, long id) -> {
Toast.makeText(this, "position=" + position, Toast.LENGTH_SHORT).show();
});

同样,只有一行语句时,可以进一步简化为:

listView.setOnItemClickListener((AdapterView<?> parent, View view, int position, long id) -> System.out.println(position));
listView.setOnItemClickListener((parent, view, position, id) -> System.out.println(position));

使用局限性

1、只适用于单一方法的接口,如果接口有多个需要实现的方法,则不能使用,如:

listView.setOnScrollListener(new AbsListView.OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
} @Override
public void onScroll(AbsListView view, int first, int count, int total) {
}
});

这是 Lambda 表达式最大的局限性

2、如果需要在匿名内部类中定义成员变量,则不能使用,如:

listView.setOnClickListener(new View.OnClickListener() {
private boolean b; @Override
public void onClick(View v) {
}
});

其实这里的成员变量没有任何意义,和定义成局部变量没什么区别

3、如果需要在匿名内部类中使用到匿名内部类的this、super,而非如外部类的this、super,则不能使用,如:

view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
//这里的this代表的是匿名内部类OnGlobalLayoutListener
view.getViewTreeObserver().removeOnGlobalLayoutListener(this);//使用到了this
}
});

Lambda 主体中的 this、super 引用与封闭上下文中一样,因为 lambda 表达式不会引入新的作用域,这与匿名类不同。

Lambda 表达式中使用的this、super是指外部类,其实往往这是 Lambda 的优点而非缺点,因为我们往往需要的就是外部类的this、super

2017-07-17

Lambda表达式 简介 语法 示例 匿名内部类的更多相关文章

  1. Lambda表达式 简介 语法 示例

    Lambda 表达式也称为闭包,是匿名类的简短形式.Lambda 表达式简化了[单一抽象方法声明接口]的使用,因此 lambda 表达式也称为功能接口. 在 Java SE 7 中,单一方法接口可使用 ...

  2. Java 8特性探究(1):通往lambda之路与 lambda表达式10个示例

    本文由 ImportNew 函数式接口 函数式接口(functional interface 也叫功能性接口,其实是同一个东西).简单来说,函数式接口是只包含一个方法的接口.比如Java标准库中的ja ...

  3. Java 8 Lambda表达式10个示例【存】

    PS:不能完全参考文章的代码,请参考这个文件http://files.cnblogs.com/files/AIThink/Test01.zip 在Java 8之前,如果想将行为传入函数,仅有的选择就是 ...

  4. java8 快速入门 lambda表达式 Java8 lambda表达式10个示例

    本文由 ImportNew - lemeilleur 翻译自 javarevisited.欢迎加入翻译小组.转载请见文末要求. Java 8 刚于几周前发布,日期是2014年3月18日,这次开创性的发 ...

  5. Java8 lambda表达式10个示例

    Java 8 刚于几周前发布,日期是2014年3月18日,这次开创性的发布在Java社区引发了不少讨论,并让大家感到激动.特性之一便是随同发布的lambda表达式,它将允许我们将行为传到函数里.在Ja ...

  6. Lambda表达式常用代码示例

    Lambda表达式常用代码示例 2017-10-24 目录 1 Lambda表达式是什么2 Lambda表达式语法3 函数式接口是什么  3.1 常用函数式接口4 Lambdas和Streams结合使 ...

  7. Java8 lambda表达式10个示例<转>

    例1.用lambda表达式实现Runnable 我开始使用Java 8时,首先做的就是使用lambda表达式替换匿名类,而实现Runnable接口是匿名类的最好示例.看一下Java 8之前的runna ...

  8. java8:(Lambda 表达式简介)

    JDK8的新特性——Lambda表达式 JDK8已经发布快4年的时间了,现在来谈它的新特性显得略微的有点“不合时宜”.尽管JDK8已不再“新”,但它的重要特性之一——Lambda表达式依然是不被大部分 ...

  9. 【Java8新特性】Lambda表达式基础语法,都在这儿了!!

    写在前面 前面积极响应读者的需求,写了两篇Java新特性的文章.有小伙伴留言说:感觉Lambda表达式很强大啊!一行代码就能够搞定那么多功能!我想学习下Lambda表达式的语法,可以吗?我的回答是:没 ...

随机推荐

  1. 【知了堂学习笔记】java 方法重载与重写的归纳

    方法重载:Java的方法重载,就是在类中可以创建多个方法,它们可以有相同的名字,但必须具有不同的参数,即或者是参数的个数不同,或者是参数的类型不同.调用方法时通过传递给它们的不同个数和类型的参数来决定 ...

  2. 【转】让你10分钟搞定Mac--最简单快速的虚拟安装

    文章出处:让你10分钟搞定Mac--最简单快速的虚拟安装http://bbs.itheima.com/thread-106643-1-1.html (出处: 黑马程序员训练营论坛) 首先说明一下. 第 ...

  3. [ 原创 ]Centos 7.0下启动 Tomcat8.5.15

    1.打开8080端口  firewall-cmd --zone=public --add-port=8080/tcp --permanent 2.重启防火墙   firewall-cmd --relo ...

  4. ARM 寄存器

    ARM总共有37个寄存器 ARM寄存器物理分类 通用寄存器:1:不分组寄存器(R0--R7) 2:分组寄存器(R8-R14) 3:程序计数器(R15)(注意:又名pc指针) 程序状态寄存器:1:CPS ...

  5. codevs 5971 打击犯罪

    5971 打击犯罪 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 某个地区有n(n<=1000)个犯罪团伙,当地警方按照他们 ...

  6. 2018 计蒜之道复赛 贝壳找房魔法师顾问(并查集+dfs判环)

    贝壳找房在遥远的传奇境外,找到了一个强大的魔法师顾问.他有 22 串数量相同的法力水晶,每个法力水晶可能有不同的颜色.为了方便起见,可以将每串法力水晶视为一个长度不大于 10^5105,字符集不大于  ...

  7. Codeforces 1090J $kmp+hash+$二分

    题意 给出两个字符串\(s\)和\(t\),设\(S\)为\(s\)的任意一个非空前缀,\(T\)为\(t\)的任意一个非空前缀,问\(S+T\)有多少种不同的可能. Solution 看了一圈,感觉 ...

  8. Codeforces Round #256 (Div. 2) E Divisors

    E. Divisors Bizon the Champion isn't just friendly, he also is a rigorous coder. Let's define functi ...

  9. Python学习笔记(三):随机生成函数方法

    本文是在Python2下总结! Python中的random模块用于生成随机数,如果想生成随机数需要先导入random的模块然后才能使用其中的方法,下面简单介绍一下常用的结果函数方法: 1·.rand ...

  10. 基于(Redis | Memcache)实现分布式互斥锁

    设计一个缓存系统,不得不要考虑的问题就是:缓存穿透.缓存击穿与失效时的雪崩效应. 缓存击穿 缓存穿透是指查询一个一定不存在的数据,由于缓存是不命中时被动写的,并且出于容错考虑,如果从存储层查不到数据则 ...