函数式接口

函数式接口(functional interface 也叫功能性接口,其实是同一个东西)。简单来说,函数式接口是只包含一个方法的接口。比如Java标准库中的java.lang.Runnable和 java.util.Comparator都是典型的函数式接口。java 8提供 @FunctionalInterface作为注解,这个注解是非必须的,只要接口符合函数式接口的标准(即只包含一个方法的接口),虚拟机会自动判断, 但 最好在接口上使用注解@FunctionalInterface进行声明,以免团队的其他人员错误地往接口中添加新的方法。
Java中的lambda无法单独出现,它需要一个函数式接口来盛放,lambda表达式方法体其实就是函数接口的实现,下面讲到语法会讲到 。

Lambda语法

包含三个部分

  1. 一个括号内用逗号分隔的形式参数,参数是函数式接口里面方法的参数

  2. 一个箭头符号:->

  3. 方法体,可以是表达式和代码块,方法体函数式接口里面方法的实现,如果是代码块,则必须用{}来包裹起来,且需要一个return 返回值,但有个例外,若函数式接口里面方法返回值是void,则无需{}

public class TestLambda {

    public static void runThreadUseLambda() {
//Runnable是一个函数接口,只包含了有个无参数的,返回void的run方法;
//所以lambda表达式左边没有参数,右边也没有return,只是单纯的打印一句话
new Thread(() ->System.out.println("lambda实现的线程")).start();
} public static void runThreadUseInnerClass() {
//这种方式就不多讲了,以前旧版本比较常见的做法
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("内部类实现的线程");
}
}).start();
} public static void main(String[] args) {
TestLambda.runThreadUseLambda();
TestLambda.runThreadUseInnerClass();
}
}

可以看出,使用lambda表达式设计的代码会更加简洁,而且还可读。

方法引用

其实是lambda表达式的一个简化写法,所引用的方法其实是lambda表达式的方法体实现,语法也很简单,左边是容器(可以是类名,实例名),中间是"::",右边是相应的方法名。

一般方法的引用格式是

  1. 如果是静态方法,则是ClassName::methodName。如 Object ::equals

  2. 如果是实例方法,则是Instance::methodName。如Object obj=new Object();obj::equals;

  3. 构造函数.则是ClassName::new

再来看一个完整的例子,方便理解

import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import javax.swing.JButton;
import javax.swing.JFrame; public class TestMethodReference { public static void main(String[] args) { JFrame frame = new JFrame();
frame.setLayout(new FlowLayout());
frame.setVisible(true); JButton button1 = new JButton("点击我!");
JButton button2 = new JButton("点击我!"); frame.getContentPane().add(button1);
frame.getContentPane().add(button2);
//这里addActionListener方法的参数是ActionListener,是一个函数式接口
//使用lambda表达式方式
button1.addActionListener(e -> { System.out.println("这里是Lambda实现方式"); });
//使用方法引用方式
button2.addActionListener(TestMethodReference::doSomething); }
/**
* 这里是函数式接口ActionListener的实现方法
* @param e
*/
public static void doSomething(ActionEvent e) { System.out.println("这里是方法引用实现方式"); }
}

为什么需要Lambda表达式

主要有三个原因:

   > 更加紧凑的代码
     比如Java中现有的匿名内部类以及监听器(listeners)和事件处理器(handlers)都显得很冗长
   > 修改方法的能力(我个人理解为代码注入,或者有点类似JavaScript中传一个回调函数给另外一个函数)
     比如Collection接口的contains方法,当且仅当传入的元素真正包含在集合中,才返回true。而假如我们想对一个字符串集合,传入一个字符串,只要这个字符串出现在集合中(忽略大小写)就返回true。
     简单地说,我们想要的是传入“一些我们自己的代码”到已有的方法中,已有的方法将会执行我们传入的代码。Lambda表达式能很好地支持这点
   > 更好地支持多核处理
     例如,通过Java 8新增的Lambda表达式,我们可以很方便地并行操作大集合,充分发挥多核CPU的潜能。
     并行处理函数如filter、map和reduce。

最新文章

  1. 【转载】Memcached在.Net中的基本操作
  2. Java获取前天和后天的时间
  3. java类加载机制
  4. SU suspike命令学习
  5. jsp导出excel
  6. [置顶] Hibernate从入门到精通(七)多对一单向关联映射
  7. Eclipse开发Maven项目提示:程序包org.junit不存在解决方案
  8. Mybatis Dynamic Query 简单筛选
  9. 自动化测试selenium(三) 由于iframe 定位不到元素
  10. (Detected problems with API compatibility(visit g.co/dev/appcompat for more info)
  11. php 添加环境变量
  12. JAVA中String.format()的使用
  13. linux系统用户和组管理
  14. SharePoint online Multilingual support - Development(2)
  15. table中head表头固定,body滚动
  16. mongodb备份策略
  17. React Native八大Demo
  18. HDOJ 5019 Revenge of GCD
  19. C语言实现可复用栈
  20. Java Zip压缩

热门文章

  1. 拒绝黑盒应用-Spring Boot 应用可视化监控
  2. Kafka权威指南阅读笔记(第六章)
  3. Prometheus 介绍详解
  4. 有Bug?你的代码神兽选对了吗
  5. 配置Docker镜像加速
  6. [Spark]Spark-streaming通过Receiver方式实时消费Kafka流程(Yarn-cluster)
  7. [py2neo]Ubuntu14 安装py2neo失败问题解决
  8. shell 字符串比较与脚本 too many arguments 报错
  9. 【PyTorch教程】P3. Python学习中的两大法宝函数(当然也可以用在PyTorch)
  10. Elastic Stack 开源的大数据解决方案