Java8 学习总结 - 基本思想

1. 行为参数化

在java8之前, 我们想给方法传递不同的行为, 最好的办法就是匿名类了:
比如下面我们想从一个装满苹果的list中的筛选出红苹果,需要传入一个 ApplePredicate, 于是我们在调用filterApples会传入一个匿名类, 实现了test方法来判断苹果的颜色是否为红色。

//过滤方法
public static List<Apple> filterApples(List<Apple> inventory, ApplePredicate p) {
  List<Apple> result = new ArrayList<>();
  for (Apple apple : inventory){
    if (p.test(apple)){
      result.add(apple);
    }
  }
  return result;
}

//接口
public interface ApplePredicate{
  boolean test (Apple apple);
}

//方法调用
List<Apple> redApples = filterApples(inventory, new ApplePredicate() { 
  public boolean test(Apple a){
    return "red".equals(a.getColor());
  }
});

其实我们需要的就是不同的test方法,但是由于在java8之前不支持Lambda表达式, 所以也需要创建一个对象来实现这个方法。
但是java8我们可以这么来完成这件事:

List<Apple> result =
        filterApples(inventory, (Apple apple) -> "red".equals(apple.getColor()));

(Apple apple) -> "red".equals(apple.getColor()) 便是在java8中可以直接传递的函数, 这种表达式叫做Lamda表达式,可能现在还看不懂这个表达, 下面我们会详细讲解下lambda表达式。

lamda表达式结构
  • 参数列表——这里它采用了 Comparator 中 compare 方法的参数,两个 Apple 。
  • 箭头——箭头 -> 把参数列表与Lambda主体分隔开。
  • Lambda主体——比较两个 Apple 的重量。表达式就是Lambda的返回值了。

在参数列表里的参数可以在后面的Lambda表达式中使用, 而后面表达式执行结束的值是默认的返回值, 也可以使用return来显式返回。

2. Lambda表达式在哪里使用?

什么条件下Lambda表达式可以作为参数传递呢? 我们需要函数式接口来作为参数。
之前我们在筛选苹果时使用了接口:

public interface ApplePredicate{
  boolean test (Apple apple);
}

这样只定义了一个抽象方法的接口便是函数式接口。 上面我们使用Lambda表达式或者使用匿名类都是对这个接口进行了实现, 原则上是一样的。

3. 函数描述符:

函数式接口的抽象方法的签名基本上就是Lambda表达式的签名。我们将这种抽象方法叫作
函数描述符
例如上面ApplePredicate的函数描述符就是 Apple -> boolean, 传入一个Apple对象返回一个boolean。

另外java提供了一个标注: @FunctionalInterface
这个标注用于表示该接口会设计成一个函数式接口。如果你用 @FunctionalInterface 定义了一个接口,而它却不是函数式接口的话,编译器将返回一个提示原因的错误。

使用函数式接口

Java8提供了几个函数式接口, 下面来分别介绍下:

Predicate
@FunctionalInterface
public interface Predicate<T>{
  boolean test(T t);
}


public static <T> List<T> filter(List<T> list, Predicate<T> p) {
  List<T> results = new ArrayList<>();
  for(T s: list){
    if(p.test(s)){
      results.add(s);
    }
  }
  return results;
}
Predicate<String> nonEmptyStringPredicate = (String s) -> !s.isEmpty();
List<String> nonEmpty = filter(listOfStrings, nonEmptyStringPredicate);

Predicate定义了一个叫test的抽象方法, 接受泛型T对象, 并返回一个boolean, 可以在过滤时使用

Consumer
@FunctionalInterface
public interface Consumer<T>{
  void accept(T t);
}


public static <T> void forEach(List<T> list, Consumer<T> c){
  for(T i: list){
    c.accept(i);
  }
}
forEach(Arrays.asList(1,2,3,4,5), (Integer i) -> System.out.println(i));

Consumer定义了一个accept的抽象方法,接受泛型T对象,不返回

Function
@FunctionalInterface
public interface Function<T, R>{
  R apply(T t);
}

public static <T, R> List<R> map(List<T> list, Function<T, R> f) {
  List<R> result = new ArrayList<>();
  for(T s: list){
    result.add(f.apply(s));
  }
  return result;
}
List<Integer> l = map(Arrays.asList("lambdas","in","action"), (String s) -> s.length());

另外还提供了一些接口

函数式接口补充

4. lambda访问局部变量

访问局部变量

你可能会问自己,为什么局部变量有这些限制。
第一,实例变量和局部变量背后的实现有一个关键不同。实例变量都存储在堆中,而局部变量则保存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程,可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问它的副本,而不是访问原始变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制。
第二,这一限制不鼓励你使用改变外部变量的典型命令式编程模式(我们会在以后的各章中解释,这种模式会阻碍很容易做到的并行处理)

闭包

5. 方法引用

如何构建方法引用
方法引用主要有三类。
(1) 指向静态方法的方法引用(例如 Integer 的 parseInt 方法,写作 Integer::parseInt )。
(2) 指 向 任 意 类 型 实 例 方 法 的 方 法 引 用 ( 例 如 String 的 length 方 法 , 写 作 String::length )
(3) 指向现有对象的实例方法的方法引用(假设你有一个局部变量 expensiveTransaction
用于存放 Transaction 类型的对象,它支持实例方法 getValue ,那么你就可以写 expensive-Transaction::getValue )

第二种和第三种方法引用可能乍看起来有点儿晕。类似于 String::length 的第二种方法引用的思想就是你在引用一个对象的方法,而这个对象本身是Lambda的一个参数。例如,Lambda表达式 (String s) -> s.toUppeCase() 可以写作 String::toUpperCase 。但第三种方法引用指 的 是 , 你 在 Lambda 中 调 用 一 个 已 经 存 在 的 外 部 对 象 中 的 方 法 。 例 如 , Lambda 表 达 式()->expensiveTransaction.getValue() 可以写作 expensiveTransaction::getValue 。

方法引用例子
方法引用实战

现在我们来尝试实现一个专门的comparator来对一个装满Apple的List进行排序

  1. 传递代码
    因为java已经有了一个sort的接口:
    void sort(Comparator<? super E> c)
    看来我们需要实现一个Comparator的接口才行, 首先我们尝试用最早的匿名类来实现:
inventory.sort(new Comparator<Apple>() {
  public int compare(Apple a1, Apple a2){
    return a1.getWeight().compareTo(a2.getWeight());
  }
});
  1. 使用lambda表达式
    匿名类的实现太啰嗦了, 我们来尝试用lambda表达式:
inventory.sort((a1, a2) -> a1.getWeight().compareTo(a2.getWeight()));
因为Comparator有一个静态方法
Comparator<Apple> c = Comparator.comparing((Apple a) -> a.getWeight());
我们就可以再精简为
inventory.sort(comparing((a) -> a.getWeight()));
  1. 使用方法引用
inventory.sort(comparing(Apple::getWeight));

6. 复合Lambda表达式

  1. 比较器复合:
inventory.sort(comparing(Apple::getWeight)
.reversed()
.thenComparing(Apple::getCountry));
  1. 谓词复合:
    谓词接口包括三个方法:negate、and和or,让你可以重用已有的Predicate来创建更复杂的谓词
Predicate<Apple> redAndHeavyAppleOrGreen = redApple.and(a -> a.getWeight() > 150).or(a -> "green".equals(a.getColor()));
  1. 函数复合:
    还可以把Function接口所代表的Lambda表达式复合起来。Function接口为此配了andThen和compose两个默认方法,它们都会返回Function的一个实例。
    Function<Integer, Integer> f = x -> x + 1;
    Function<Integer, Integer> g = x -> x * 2;
    用compose的话,它将意味着f(g(x)),而andThen则意味着g(f(x))
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,367评论 6 512
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,959评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,750评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,226评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,252评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,975评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,592评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,497评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,027评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,147评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,274评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,953评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,623评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,143评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,260评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,607评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,271评论 2 358

推荐阅读更多精彩内容

  • 简介 概念 Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主...
    刘涤生阅读 3,207评论 5 18
  • Java8 in action 没有共享的可变数据,将方法和函数即代码传递给其他方法的能力就是我们平常所说的函数式...
    铁牛很铁阅读 1,235评论 1 2
  • 第一章 为什么要关心Java 8 使用Stream库来选择最佳低级执行机制可以避免使用Synchronized(同...
    谢随安阅读 1,493评论 0 4
  • 注:之前关于Java8的认知一直停留在知道有哪些修改和新的API上,对Lambda的认识也是仅仅限于对匿名内部类的...
    mualex阅读 2,826评论 1 4
  • lambda表达式(又被成为“闭包”或“匿名方法”)方法引用和构造方法引用扩展的目标类型和类型推导接口中的默认方法...
    183207efd207阅读 1,486评论 0 5