lambda表达式

目的:行为参数化

Lambda表达式是简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表。

Lambda的基本语法是(parameters) -> expression(parameters) -> { statements; }。其中, (parameters) -> expression 的表达式中隐含了return,如 () -> 42 (parameters) -> { statements; } 的花括号内是语句。

举例:

() -> 42   //参数为空,返回一个int
(List<String> list) -> list.isEmpty() //参数为list,返回一个boolean
(int x, int y) -> x*y //参数为两个int,返回一个int
(String s) -> System.out.println(s); //参数为一个String,不返回结果
(String s) -> {System.out.println(s);} //参数为一个String,打印字符串

哪些地方使用哪些lambda

函数式接口是只定义一个抽象方法的接口,即使拥有多个默认方法。FunctionalInterface 标注一个函数式接口,会加入编译检查。函数式接口中默认方法的目的是:改变已发布的接口而不破坏已有的实现。

在接受函数式接口为参数的地方,都可以使用lambda表达式。

栗子:

public void execute(Runnable r){
r.run();
}
execute(() -> {}); //使用lambda,Runnable是参数为空,没有返回值的函数式接口,即() -> void //fetch返回一个函数式接口,() -> String
public Callable<String> fetch() {
return () -> "Tricky example ;-)";
}

喵:为什么只有在函数式接口的地方使用呢?lambda表达式没有函数名,只有参数列表,函数主体和返回值,如果接口有多个方法,就不能直接匹配到正确的方法上了,所以,只有一个抽象方法的函数式接口可以满足。

Predicate

java.util.function.Predicate<T>是一个含有多个默认方法的函数式接口,抽象方法为:(T t) -> bool。看下代码,你就懂了~

FunctionalInterface
public interface Predicate<T> {
//接口方法,入参为泛型T,返回bool。即:(T t) -> bool
boolean test(T t);
//默认方法,and操作
default Predicate<T> and(Predicate<? super T> other) {
Objects.requireNonNull(other);
return (t) -> test(t) && other.test(t);
}
//默认方法,取反操作
default Predicate<T> negate() {
return (t) -> !test(t);
}
//默认方法,or 操作
default Predicate<T> or(Predicate<? super T> other) {
Objects.requireNonNull(other);
return (t) -> test(t) || other.test(t);
}
//默认方法,判断是否相等
static <T> Predicate<T> isEqual(Object targetRef) {
return (null == targetRef)
? Objects::isNull
: object -> targetRef.equals(object);
}

使用

Predicate<String> nonEmptyStringPredicate = (String s) -> !s.isEmpty();

Consumer

java.util.function.Consumer<T> 是一个只含有一个默认方法的函数式接口,抽象方法为:(T t) ->void。看下代码,你就懂了~

@FunctionalInterface
public interface Consumer<T> {
//接口方法,入参为泛型T,返回void。即:(T t) -> void
void accept(T t); //默认方法,可以执行级联操作
default Consumer<T> andThen(Consumer<? super T> after) {
Objects.requireNonNull(after);
return (T t) -> { accept(t); after.accept(t); };
}

Function

java.util.function.Function<T, R>是一个含有多个默认方法的函数式接口,抽象方法为:(T t) ->R。看下代码,你就懂了~

@FunctionalInterface
public interface Function<T, R> {
//接口方法,入参为泛型T,返回泛型R。即:(T t) -> R
R apply(T t);
//默认方法,实现级联操作。before方法输入V,输出T,本function输入T,输出R。
default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
Objects.requireNonNull(before);
return (V v) -> apply(before.apply(v));
}
//默认方法,级联操作
default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
Objects.requireNonNull(after);
return (T t) -> after.apply(apply(t));
}
//默认方法,输入啥,输出啥
static <T> Function<T, T> identity() {
return t -> t;
}

特定方法避免装箱操作

在处理数据时,使用特定方法,可以避免装箱操作,如:IntPredicate、LongConsumer、DoubleFunction等。具体见API库。

总结

函数描述符 函数式接口
(T) ->bool java.util.function.Predicate<T>
(T) -> void java.util.function.Consumer<T>
(T) -> R java.util.function.Function<T, R>
(T,U) -> R java.util.function.BiFunction<T, U, R>
() -> T java.util.function.Supplier<T>

只要函数描述符兼容,函数式接口就可以复用。

特殊的void兼容规则:

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

方法引用

方法引用是调用单一方法的Lambda的快捷写法,格式ClassName::methodName。看下栗子你就懂了~

一般方法引用

(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
(list, element) -> list.contains(element) 等价于 List::contains

主要有三类:

  • 指向静态方法的方法引用,如:Integer::parseInt
  • 指向任意类型实例方法的方法引用,如:String::length
  • 指向现有对象的实例方法的方法引用,如:User::getUserId

构造函数引用

  • 无参构造函数

    无参构造函数的函数描述符: () -> T,由上面的总结知,可以使用Supplier接口,如下:
Supplier<User> c1 = User::new;  // c1 = () -> new User();
User user = c1.get();
  • 有一个参数的构造函数

    有一个参数的构造函数的函数描述符是(T) -> R,可以使用Function接口,如下:
Function<Long, User> c2 = User::new;
User user = c2.apply(110L);
  • 有三个参数的构造函数

    有三个参数的构造函数的函数描述符是(T,U,V) -> R,没有现成的接口,需要自定义,如下:
@Data
@AllArgsConstructor
public class User {
private String name;
private Long userId;
private Integer age;
} @FunctionalInterface
public interface TriFunction<T,U,V,R> {
R create(T t, U u, V v);
} public static void main(String[] args) {
TriFunction<String, Long, Integer, User> triFunction = User::new;
User user = triFunction.create("tina", 12L, 13);
}

使用注意事项

  • Lambda表达式可以引用静态变量、成员变量和最终的(final) 或事实上最终的局部变量。

java 8 总结 之lambda的更多相关文章

  1. Java函数式编程和lambda表达式

    为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...

  2. Java核心技术-接口、lambda表达式与内部类

    本章将主要介绍: 接口技术:主要用来描述类具有什么功能,而并不给出每个功能的具体实现.一个类可以实现一个或多个接口. lambda表达式:这是一种表示可以在将来的某个时间点执行的代码块的简洁方法. 内 ...

  3. Java基础教程:Lambda表达式

    Java基础教程:Lambda表达式 本文部分内容引用自OneAPM:http://blog.oneapm.com/apm-tech/226.html 引入Lambda Java 是一流的面向对象语言 ...

  4. java函数式编程之lambda表达式

    作为比较老牌的面向对象的编程语言java,在对函数式编程的支持上一直不温不火. 认为面向对象式编程就应该纯粹的面向对象,于是经常看到这样的写法:如果你想写一个方法,那么就必须把它放到一个类里面,然后n ...

  5. 转载java 8 为什么引入 lambda

    转载:https://www.cnblogs.com/keeya/p/11404631.html 在Java8出现之前,如果你想传递一段代码到另一个方法里是很不方便的.你几乎不可能将代码块到处传递,因 ...

  6. 最全最强 Java 8 - 函数编程(lambda表达式)

    Java 8 - 函数编程(lambda表达式) 我们关心的是如何写出好代码,而不是符合函数编程风格的代码. @pdai Java 8 - 函数编程(lambda表达式) 简介 lambda表达式 分 ...

  7. Java 8:掌握 Lambda 表达式

    本文将介绍 Java 8 新增的 Lambda 表达式,包括 Lambda 表达式的常见用法以及方法引用的用法,并对 Lambda 表达式的原理进行分析,最后对 Lambda 表达式的优缺点进行一个总 ...

  8. [Java 8] (10) 使用Lambda完成函数组合,Map-Reduce以及并行化

    好文推荐!!!!! 原文见:http://blog.csdn.net/dm_vincent/article/details/40856569 Java 8中同时存在面向对象编程(OOP)和函数式编程( ...

  9. 0028 Java学习笔记-面向对象-Lambda表达式

    匿名内部类与Lambda表达式示例 下面代码来源于:0027 Java学习笔记-面向对象-(非静态.静态.局部.匿名)内部类 package testpack; public class Test1{ ...

  10. Upgrading to Java 8——第一章 Lambda表达式

    第一章 Lambda表达式 Lamada 表达式是Java SE 8中最重要的新特性,长期以来被认为是在Java中缺失的特性,它的出现使整个java 语言变得完整.至少到目前,在这节中你将学习到什么是 ...

随机推荐

  1. Android学习笔记-ImageView(图像视图)

    本节引言: 本节介绍的UI基础控件是:ImageView(图像视图),见名知意,就是用来显示图像的一个View或者说控件! 官方API:ImageView;本节讲解的内容如下: ImageView的s ...

  2. OpenLayers3--ol3--新特性

    OL3: A high-performance, feature-packed library for all your mapping needs < 一个可以满足各种地图应用的高性能的.功能 ...

  3. NYOJ--257--郁闷的C小加(一)(中缀表达式变后缀表达式 )

    郁闷的C小加(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 我们熟悉的表达式如a+b.a+b*(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说 ...

  4. Mongodb启动&关闭

    mac 下mongo的启动和关闭以及启动问题解决 mongo的安装在这:http://www.cnblogs.com/leinov/p/6855784.html Mac os mongodb数据安装路 ...

  5. Python查询SQLserver数据库备份(抛砖引玉)

    通过python pymssql直接访问SQLserver数据库,查找其数据库mode,这个脚本具有很强的抛砖引玉特性: 1.可以巡检多台多数据库服务器 2.query内容可以多样化,譬如查询死锁.连 ...

  6. [NOIP2015] 斗地主 大爆搜

    考试的时候想了半天,实在是想不到解决的办法,感觉只能暴力..然后暴力也懒得打了,小数据模拟骗30分hhh 然而正解真的是暴力..大爆搜.. 然后我的内心拒绝改这道题(TAT) 不过在wcx大佬的帮助下 ...

  7. IDE eclipse PyDev插件安装

    Python安装成功后,即要配置开发环境,这里选用Eclipse, 在Eclipse中安装PyDev插件,有多种方法,这里介绍最最常用的两种. 1)使用Eclipse安装插件,打开eclipse,进入 ...

  8. PL/SQL 编程(二)游标、存储过程、函数

    游标--数据的缓存区 游标:类似集合,可以让用户像操作数组一样操作查询出来的数据集,实质上,它提供了一种从集合性质的结果中提取单条记录的手段. 可以将游标形象的看成一个变动的光标,他实质上是一个指针, ...

  9. Webservice 中涉及的几个概念

    开发webservice有一段时间了.但是对其中的协议及实现一直存在疑惑.这里搜集相关概念.用以更好的理解它. 一.什么是Webservice WebService是一种跨编程语言和跨操作系统平台的远 ...

  10. HDU 1176 免费馅饼:dp

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1176 题意: 横坐标范围为[0,10],你在第0秒站在坐标为5的地方. 在接下来的一段时间内,会有n个 ...