Java8新特性之:Lambda表达式

一. Lambda定义(λ):

-- 匿名,它不像普通方法那样有一个明确的名称;

-- 函数,它不像普通方法那样属于某个特定的类,但和方法一样,Lambda有参数列表、函数主体、返回类型或抛出异常列表:

-- 传递,Lambda可以作为参数传递给方法或存储在变量中:

-- 简洁。

二. Lambda表达式结构:

1. 参数列表;

2. 箭头:箭头->把参数列表与Lambda主体分隔开;

3. Lambda主体:表达式就是Lambda表达式的例子。

三.Lambda基本语法:

(parameters) -> expression

(parameters) -> { statements; }

使用显式返回语句时需要使用花括号“{}”。

eg:  Lambda示例:

使用案例 Lambda示例
无参数,返回void () -> {}
无参数,返回String () -> "Raoul"
无参数,返回String(利用显式返回语句) () -> { return "Result";}
布尔表达式 (List<String> list) -> list.isEmpty()
创建对象 () -> new Apple(10);
消费一个对象 (Apple apple) -> { System.out.println(a.getWeight()); }
从一个对象中选择/抽取 (String s) -> s.length()
组合两个值 (int a, int b) -> a * b
比较两个对象 (Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight())

四. 在哪里使用Lambda

可以在函数式接口上使用Lambda表达式。

函数式接口:只定义一个抽象方法的接口。

@FunctionalInterface:表明为函数式接口,但它不是必须的。

(T, U) -> R表达式展示了应当如何思考函数描述符。左侧代表了参数类型。这里它代表一个函数,具有两个参数,分别为泛型T和U,返回类型为R。

Java的泛型只能绑定到引用类型,当需要引用原始类型时因为自动装箱和拆箱机制时,装箱后的值需要更多的内存,需要付出性能代价,为了避免装箱操作对Predicate<T>和Function<T, R>等通用函数式接口的原始类型特化:IntPredicate、IntToLongFunction等。

Java8中的常用函数式接口:

函数式接口 函数描述符 原始类型特化
Predicate<T> T -> boolean
IntPredicate,

LongPredicate,

DoublePredicate

Consumer<T> T -> void
IntConsumer,

LongConsumer,

DoubleConsumer

Function<T, R> T -> R
IntFunction<R>,

IntToDoubleFunction,

IntToLongFunction,

LongFunction<R>,

LongToDoubleFunction,

LongToIntFunction,

DoubleFunction<R>,

ToIntFunction<T>,

ToDoubleFunction<T>,

ToLongFunction<T>

Supplier<T> () -> T
BooleanSupplier,

IntSupplier,

LongSupplier,

DoubleSupplier

UnaryOperator<T> T -> T
IntUnaryOperator,

LongUnaryOperator,

DoubleUnaryOperator

BinaryOperator<T> (T, T) -> T
IntBinaryOperator,

LongBinaryOperator,

DoubleBinaryOperator

BiPredicate<L, R> (L, R) -> boolean
BiConsumer<T, U> (T, U) -> void
ObjectIntConsumer<T>,

ObjectLongConsumer<T>,

ObjectDoubleConsumer<T>

BiFunction<T, U, R> (T, U) -> R
ToIntBiFunction<T, U>,

ToLongBiFunction<T, U>,

ToDoubleBiFunction<T, U>

Runnable () -> void

当需要Lambda表达式抛出异常时,有两种方式:

-- 自己编写新的函数式接口,并声明受检异常(任何函数式接口都不允许抛出受检异常);

-- 将Lambda包在一个try/catch块中。

@FunctionalInterface
public interface BufferedReaderProcessor {
  String process(BufferedReader b) throws IOException;
}
Function<BufferedReader, String> f = (BufferedReader b) -> {
  try {   
    return b.readLine();
  } catch (IOException e) {
    throw new RuntimeException(e);
  }
};

当Lambda表达式抛出一个异常时,throws语句也必须与Lambda所指类型相匹配。

如果Lambda的主体是一个语句表达式,它就和一个返回void的函数描述符兼容(当然需要参数列表也兼容)。

eg:尽管List的add方法返回的是boolean,但以下两行都是合法的:

//Predicate返回了一个boolean
Predicate<String> p = s -> list.add(s);

//Consumer返回了一个void
Consumer<String> b = s -> list.add(s);

Lambda类型推断:Java编译器会从上下文(目标类型)中推断出用什么函数式接口来配合Lambda表达式,所以也能推断出适合Lambda的签名,因为函数描述符可以通过目标类型来得到。这样就可以在Lambda中省去标注参数类型,当参数只有一个时还可以省去参数的括号。

Lambda使用局部变量的限制:Lambda表达式对值封闭,而不是对变量封闭。Lambda表达式引用的局部变量必须是final的,只能有一次赋值。

这是因为实例变量是储存在堆中,而局部变量是储存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问他的副本而不是访问原始变量。

Lambda方法引用:

方法引用就是Lambda的快捷写法。目标引用放在分隔符::前面,方法的名称放在后面。

eg:

Lambda表达式 等效的方法引用
(Apple a) -> a.getWeight() Apple::getWeight
() -> Thread.currentThread().dumpStack() Thread.currentThread()::dumpStack
(str, i) -> str.substring(i) String::substring
(String s) -> System.out.println(s) System.out::println

如何构建方法引用:

主要有三类:

-- 指向静态方法的放方法引用;

-- 指向任意类型实例方法的方法引用;

-- 指向现有对象的实例方法的方法引用。

Lambda构造函数引用:

对于一个现有构造函数,可以利用它的名称和关键字new来创建它的一个引用:ClassName::new。它的功能与静态方法的引用类似。

Lambda表达式 等价构造方法引用Lambda表达式

Supplier<Apple> c1 = Apple::new;

Apple a1 = c1.get();


Supplier<Apple> c1 = () -> new Apple();

Apple a1 = c1.get();


Function<Integer, Apple> c2 = Apple::new;

Apple a2 = c2.apply(110);


Function<Integer, Apple> c2 = (weight) -> new Apple(weight);

Apple a2 = c2.apply(110);


BiFunction<String, Integer, Apple> c2 = Apple::new;

Apple a2 = c2.apply("green", 110);


BiFunction<String, Integer, Apple> c2 = (color, weight) -> new Apple(color, weight);

Apple a2 = c2.apply("green", 110);

复合Lambda表达式的有用方法:

复合类型 方法 说明 举例
比较器复合 reversed() 逆序
//按重量递减排序

inventory.sort(comparing(Apple::getWeight))

.reversed();

thenComparing 比较器链(接受一个函数作为参数,如果两个对象用第一个Comparator比较之后是一样的,就提供第二个Comparator
//两个苹果一样重时按国家排序

inventory.sort(comparing(Apple::getWeight))

.reversed().

thenComparing(Apple::getCountry);


谓词复合

(and和or方法的优先级是按照在表达式链中的位置,从左向右确定的)

negate
//产生现有对象redApple的非

Predicate<Apple> notRedApple = redApple.negate();

and 将两个Lambda用and组合起来
//一个苹果既是红色又比较重(链接两个谓词来生成一个Predicate对象)

Predicate<Apple> RedAndHeavyApple = redApple.and(a -> a.getWeight() > 150);

or 要么
//要么重(150g以上)的红苹果,要么是绿苹果

Predicate<Apple> RedAndHeavyApple = redApple.and(a -> a.getWeight() > 150)

.or(a -> "green".equals(a.getColor()));

函数复合 andThen 先对输入应用一个给定函数,再对输出应用另一个函数
//等同于数学上的g(f(x)),返回4

Function<Integer, Integer> f = x -> x + 1;

Function<Integer, Integer> g = x -> x * 2;

Function<Integer, Integer> h = f.andThen(g);

int result = h.apply(1);

compose 先把给定的函数用作compose的参数里面给的那个函数,然后再把函数本身用于结果
//等同于数学上的f(g(x)),返回3

Function<Integer, Integer> f = x -> x + 1;

Function<Integer, Integer> g = x -> x * 2;

Function<Integer, Integer> h = f.compose(g);

int result = h.apply(1);

原文地址:http://blog.51cto.com/turnsole/2093177

时间: 2024-10-14 04:24:14

Java8新特性之:Lambda表达式的相关文章

java8新特性之——lambda表达式的使用

lambda表达式简介 个人理解,lambda表达式就是一种新的语法,没有什么新奇的,简化了开发者的编码,其实底层还是一些常规的代码.Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递).可以写出更简洁.更灵活的代码.作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升. Lambda表达式的语法(记得要在jdk1.8及以上的jdk环境下实验) Lambda 表达式的基础语法:Java8中引入了一个新的操作符 "->&

好程序员分享java8新特性之Lambda表达式

?.Lambda表达式简介 什么是Lambda? Lambda表达式是Java 8推出的?个新特性.从本质上讲,Lambda表达式是?个匿名函数. 为什么要使?Lambda? 使?Lambda表达式可以对?个接?进??常简洁的实现. 之前我们在给?个接?引?赋值的时候,可以使?接?实现类,或者匿名内部类.但是有了 Lambda表达式,我们可以更加?便的实现这个需求. interface Comparator<T>?{ int compare(T o1, T o2); } class Progr

2020了你还不会Java8新特性?Lambda表达式及API(二)

lambda表达式 为什么要使用lambda表示式 在Java中无法将函数座位参数传递给一个方法,也无法返回一个函数的方法. 在js中,函数的参数是一个函数.返回值是另一个函数的情况是非常常见的.是一门经典的函数式语言. Java匿名内部类. 匿名内部类的介绍 Gradle的使用.可以完全使用maven的中央仓库. 进行安卓的开发时,gradle已经成为标配了. lambda: 匿名内部类 my_jButton.addActionListener(new ActionListener() { @

Java8新特性03 Lambda表达式

一. Lambda表达式概述 Lambda表达式是Java8中最大的变化.它允许我们将一个函数当作方法的参数,或者说把一段代码当作数据使用.很多基于JVM平台的语言一开始就支持Lambda表达式,比如Scala,但是Java语言一直只能使用匿名内部类来替代Lambda表达式,直到Java8才使用Lambda表达式. 二.定义 Lambda表达式是一个匿名函数,它是一个函数式接口的具体实现方式:使用Lambda语法可以用来代替Java8之前的匿名内部类,从而简化代码. 任意只包含一个抽象方法的接口

Java8新特性之lambda表达式

lambda表达式 简化实例化函数式接口(functional interface)的过程 lambda表达式的优势 简化代码的实现 lambda表达式的语法 一个 Lambda 表达式可以有零个或多个参数 参数的类型既可以明确声明,也可以根据上下文来推断.例如:(int a)与(a)效果相同 所有参数需包含在圆括号内,参数之间用逗号相隔.例如:(a, b) 或 (int a, int b) 或 (String a, int b, float c) 空圆括号代表参数集为空.例如:() -> 42

C++11 新特性之 Lambda表达式

lambda表达式可以用于创建并定义匿名的函数对象,以简化编程工作 Lambda的语法如下: [函数对象参数](操作符重载函数参数)->返回值类型{函数体} []内的参数指的是Lambda表达式可以取得的变量.(2)函数中的param就是指函数可以得到在Lambda表达式外的全局变量, 如果在[]中传入=的话,即是可以取得所有的外部变量,如(1)和(3)Lambda表达式 ()内的参数是每次调用函数时传入的参数. ->后加上的是Lambda表达式返回值的类型,如(3)中返回了一个int类型的变

C++11新特性(3) lambda表达式(1)

C++11添加了一项名为lambda表达式的新功能.通过这项功能能编写内嵌的匿名函数,而不必编写独立函数或函数对象,使得代码更加理解. lambda表达式包含以下部分. [capture_block](parameters) mutable exception_specification->return_type {body} 现在分析各个部分的内容: (capture_block)捕捉块:指定如何捕捉所在作用域的变量,并供给lambda主体部分使用. (parameter)参数(可选):lam

JDK1.8新特性(二): Lambda表达式 (参数列表) -&gt; { } 和函数式接口@FunctionalInterface

Lambda表达式 二:简介 JDK的升级的目的有以下几个:增加新的功能.修复bug.性能优化.简化代码等几个方面,Lambda表达式就是属于简化代码,用于简化匿名实现类,提供一种更加简洁的写法.Lambda表达式在Swift语言中称之为代码块,Lambda表达式可以认为是一种特殊的接口,该接口必须只有一个抽象方法. 语法 (参数类型 参数名, 数参数类型 参数名2...) -> { // code }; 小括号()中的内容就是方法中的参数列表包括参数类型.参数名,其中参数类型是可以省略的,当参

Java8新特性之lambda

本系列文章翻译自@shekhargulati的java8-the-missing-tutorial Java8中最重要的特性之一就是引入了lambda表达式.这能够使你的代码更加简练,并允许你将行为传递到各处.一段时间以来,Java因为自身的冗长和缺少函数式编程的能力而受到批评.随着函数式编程变得越来越流行和有价值,Java也在努力接受函数式编程.否则,Java将会变得没有价值. Java8在使世界上最受欢迎的编程语言之一在接纳函数式编程的过程中向前迈了一大步.为了支持函数式编程,编程语言必须将

java8新增特性之lambda表达式

java 1.8 为了顺应函数式编程的大势,简化编码工作量,新增加了lambda 表达式. 由于在java中,函数并不是一等公民,不能独立存在,所以,在java中lambda 也必须依托对象存在,其用得最多的地方,是简化匿名内部类的写法. 1 public class LambdaTest { 2 3 private static MathOperation sayHi = name -> { 4 System.out.println("hi!, "+ name); 5 }; 6