Java 8之lambda表达式(一)

1.1 为什么要使用lambda表达式

"lambda 表达式"是一段可以传递的代码,因此它可以被执行一次或多次。下面让我们来下面的代码示例:

class Worker implements Runnable{
  public void run(){
      for(int i=0;i<1000;i++){
        doWork();
     }
  }  
...
}

然后直接启动一个线程
Worker worker=new Worker();
new Thread(worker).start();

你会看到,一段代码会被传递给其他调用者,也许是一个线程又或者是一个按钮,这段代码会稍后被调用。 到现在为止,在Java中传递一段代码并不容易,你不可能将代码块到处传递。由于Java是一个面向对象语言,你不得不构建一个属于某个类的对象,由它的方法来包含所需的方法。 虽然我们已经通过类、对象的方式在Java中实现相似的功能,但是这使用起来并不让人轻松和愉快。

1.2 lambda表达式的语法

lambda表达式在Java中的语法 : 1、参数 2、箭头(->) 3、表达式 如果负责计算的代码无法用一个表达式表示,那么可以用编写方法的方式来编写:即用{}包裹代码并明确使用return语句,例如:

(String first,String second)->{
  if(first.length<second.length) return -1;
    else if(first.length>second.length) return 1;
      else return 0;
}

如果lambda表达式没有参数,你仍可以提供一对小括号(),如果不含参数的方法那样:

()->{ for(int i=0;i<1000;i++) doWork();  }

如果一个lambda表达式的参数类型是可以被推导的,那么就可以省略它们的类型,例如:

Comparator<String>com=(first,second)//同(String first,String second)一样
                       ->Integer.compare(first.length(),second.length());
这里编译器会推导出first和second必须是字符串,因为lambda表达式被赋给了一个字符串比较器。

如果某个方法只含有一个参数,并且该参数的类型也可以被推导出来,你甚至可以省略小括号:

EventHandler<ActionEvent>listener=event-> Sysout.out.println("Thanks for clicking);

注意:你可以像对待方法参数一样向lambda表达式的参数添加注解或者final修饰符,如下。 (final String name)->... (@NonNull String name)->...

永远不需要为一个lambda表达式执行返回类型,它总是会从上下文中被推导出来。例如,表达式

(String frist,String second)->Integer.compare(first.length(),second.length());

可以被使用在期望结果类型为int的上下文中。

注意:在lambda表达式中,只在某些分支中返回值(其它分支没有返回值)是不合法的。例如,(int x)->{ if(x>0) return1; }是不合法的。

1.3 函数式接口

定义:只包含一个抽象方法的接口,被称之为函数式接口。

注意;你可能奇怪为什么函数式接口必须只有一个抽象方法。难道接口中的方法不都是抽象的吗?事实上,接口会经常重新声明Object类方法,例如toString或clone。而这些方法声明并不是抽象的。(Java API中的某些接口重新声明Object类方法,是为了关联javadoc的注释。具体例子可以参考Comparator API)。之后你将会看到Java 8中接口可以声明非抽象方法。

为了演示函数式接口,我们以Arrays.sort方法为例,该方法的第二个参数,需要一个Comparator(该接口只含有一个方法)的实例,接下来,我们来编写一个简单的lambda表达式:

Arrays.sort(words,(first,second)->Integer.compare(first.length(),second.length()));

你最好将lambda表达式想象成一个函数,而不是一个对象,并记住它可以被转换成一个函数式接口。

注意:你甚至不能将一个lambda表达式赋值给一个Object类型的变量,因为Object不是一个函数式接口。

请记住,任何一个lambda表达式都可以等价转换成现在所使用的API中对应的函数式接口。

注意:你可以在任意函数接口上标注@FunctionalInterface注解,这样的好处有2个。首先编译器会检查标注该注解的实体,检查它是否只包含一个抽象方法的接口。另外,在javadoc页面也会包含一条声明,说明这个接口是一个函数式接口。该注解不要求强制使用,从概念上来讲,所有只含有一个抽象方法的接口都是函数式接口,但是使用@FunctionalInterface注解会让你代码看上去更清楚。

最后,当一个lambda表达式被转换为一个函数式接口的实例时,请注意检查期异常。如果lambda表达式中可能会抛出一个检查期异常,那么该异常需要在目标接口的抽象方法中声明。例如,以下表达式会产生一个错误:

Runnable sleeper=()->{ Sysout.out.println("zzz"); Thread.sleep(1000);};
//错误:Thread.sleep可以抛出一个检查期的InterruptedException。

由于Runnable.run不能抛出任何异常,所以这个赋值是不合法的,有两种方式可以修正该问题。 ① 是在lambda表达式中捕获该异常; ② 将lambda表达式赋给一个其抽象方法可以抛出异常的接口。 例如:Callable接口的call方法可以抛出任何异常,因此,你可以将lambda表达式赋给Callable<Void>(如果你添加一条" return null " 语句)。

好了,lambda表达式的基本形式、基本概念,到这里就结束了。

接下来咱们会继续了解lambda表达式的以下内容: ① 方法引用 ② 构造器引用 ③ 变量作用域

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏锦小年的博客

Python学习笔记3.2-python内置函数大全

学习python不可避免的首先要了解python的内置函数,熟悉了这些以后可以给编程带来很大的方便。 1、数学运算类 函数名 函数功能 备注 abs...

2379
来自专栏技术博文

js 六种数据类型的区别及bool 转换判断

一、bool型转换判断: 1、true 和 1 比较是相同,false 和 0 比较是相同(是 “==” 比较),因为内部会实现数据类型的 转化,将true 转...

3165
来自专栏机器学习算法与Python学习

python: 函数详解

def square(x): return x**2 >>> square <function square at 0x031AA230> >>> dir(sq...

4697
来自专栏desperate633

深入理解javascript中的继承机制(4)多继承寄生式继承借用构造函数借用构造函数并且复制原型以上

我们知道多继承是面向对象的语言中比较纠结的一个问题,有好处也存在缺陷。这方面我们不多讨论。就javascript而言,要实现多继承是比较简单的,因为javasc...

771
来自专栏web前端

JavaScript基础学习--14 json、数组

Demos:   https://github.com/jiangheyan/JavaScriptBase 一、json      1、格式与取值:{key:...

22610
来自专栏racaljk

正则表达式

\ 将下一个字符标记为一个特殊字符、或一个原义字符、或一个向后引用、或一个八进制转义符。例如,“\n”匹配字符“n”。“\\n”匹配一个换行符。序列“\\...

1005
来自专栏测试开发架构之路

C++之类和对象的使用(二)

析构函数 析构函数的作用并不是删除对象,而是在撤销对象占用的内存之前完成一系列清理工作,使这部分内存可以被程序分配给新对象使用。对象生命周期结束,程序就自动执...

2777
来自专栏黄Java的地盘

正则表达式之入门篇

本文主要通过对正则表达式的语法进行一些简单的介绍,从而让没有接触过或者想学习正则表达式的同学有一个基础的了解,从而能够看懂和编写使用一般的正则表达式。

1131
来自专栏desperate633

详解javascript中的即时函数,内部函数,能重写自身的函数即时函数内部函数返回函数的函数能重写自己的函数小结

在上篇谈到匿名函数和回调函数的基础上,我们接着介绍javascript中的即时函数,内部函数,返回函数的函数,能重写自身的函数等几种常见的函数类型及使用方法。只...

821
来自专栏java初学

5.7 (2)反射举例

37612

扫码关注云+社区