我也说说java8的新特性

本文地址:http://www.jianshu.com/p/dd24738d2b11

写在前面:

Java8已经发布很久了,但是安卓官方对它的支持一直不是很完整,直到最近发布的Android Studio 3预览版才支持了函数式接口,集合的聚合操作等,所以现在有必要真正地来去学习它了。
本文主要针对Android Studio(3.0 canary1)对java api支持的部分做说明,当然,这部分的特性也完全是java8的一些特性,是其一个子集而没有其它差异之处。
本来这篇东西涉及到了java8的好几个特性,包括函数式接口,一些标准的函数式接口,集合的聚合操作和lambda表达式,但是一通了解准备下来,发现它们都是围绕着函数式接口这一个东西在呈现的,并且内容并不复杂,所以我打算用这一篇把这几个特性都讲了吧。当然,我这里只是挑最重要最核心最能让大家明白的几点来讲,需要更详细了解更多用法和细节的可以参看文末的官方文档地址。

既然说是围绕函数式接口来呈现的,那我们就先从函数式接口开始说起吧。

函数式接口

定义:

只有一个抽象方法的接口

( 当然,这是我的白话定义,删繁就简我们就不啰嗦它的那些default方法和静态方法了。这两个特性完全只是对接口的一些能力扩展,不必大惊小怪,所以也是为什么我觉得不需要另外一篇文章介绍这些特性。)

这个定义就是专门为了lambda表达式的应用而设计的。只要理解到这里就可以了,事实也就这些吧。它并没有新的东西。倒是api里创建了一堆标准的函数式接口给各位使用,就是下面讲的这个:

标准函数式接口

java8硬是要提供这么一个“工具类”(嗯,因为它放在了java.util.function包下,我们确实完全可以理解成是一个工具),我想也是极力引导大家运用依赖倒置的原则吧,啥扩展代码都不要写进原有方法里面,只使用这些标准接口对接相应的输入输出类型,原有方法就只调用这些接口……但这是不是想多了啊?我们需要扩展的时候,自然会自己定义一个合理的(起码名字更好理解的)接口来解耦代码,真需要你提供这么一堆?但既然提供了,我们也只好试着将就着勉强着用用先吧,起码在输入输出类型一致时,我们不需要去定义多个功能相同的接口吧(可能设计者就只是这么想呢。。。确实,就是为了省点代码?作为jdk,这是不是管得太宽想得太多了?这是不是过度设计了?为了省点代码,让api变累赘?成功与否,看大家接受程度如何吧!记忆这些接口是一回事,我个人是推崇更自由地编写代码的,包括思想上的自由,不是由你来给我设计接口,当然如果自己不会设计接口,那是水平问题,这门槛也不高,不应该由jdk来做约束,限制了会设计接口的人的自由。——这有点像摇号,看似解决问题,但却伤害着所有人的根本利益。当然,你会说你也还可以自己定义接口啊。但这不一样,你既然提供了一个这样的东西,你对整个环境的影响是在的,我定义接口时就不得不考虑那些使用这些标准接口的人,甚至说不定有些团队还会因此而将标准接口列入编码规范,这些影响都是客观的,事物都是相互联系相互影响的。扯远了,不过也可以帮助大家理解这个函数式接口吧)

java8内置的几个标准函数式接口:

Predicate

boolean test(T)
表示判断。输入一个对象,返回布尔值

Consumer

void accept(T)
消费一个操作。输入一个对象,处理(消费)完后不返回值

Supplier

T get()
直接返回一个对象

Function

R apply(T)
就是函数映射的意思,一个输入,一个相应的输出

主要是以上4种接口了,另外还有很多都是在这几种的基础之上的扩展,比如加前缀Bi的表示输入两个参数,加了类型Int的表示返回的类型用int类型等等。

又一个题外话——
这样只需要在使用的时候写实现, 类不需要再变。但省去了很多信息,是否真的有必要这样?更何况也不是所有函数都需要扩展,这样是不是太过了?原有的类是不变了,但扩展时时都要变呢。

Lambda表达式

作用

在方法中只有一条语句时,省略更多代码。

语法

1.包括一个箭头->
2.箭头左边是参数,可以各种省略类型和括号
3.箭头右边是函数体,可以各种省略花括号和关键字return

使用场景

实际上,lambda表达式只能在函数式接口上使用。 就是为了在只有一个方法时省点代码。

另外,因为它看起来很像一个函数,所以你可以当它是一个匿名方法----没有名字的方法 (来自于官方网址的说明哦)

下面,举例说明一下lambda表达式在各种具体场景中的运用吧:

  • 函数式接口中:
setOnClickListener(view->view.setAlpha(255));

相当于

setOnClickListener(new OnClickListener(){
    public void onClick(View view){
        view.setAlpha(255);
    }
})
  • 标准函数式接口也一样,只不过接口是由jdk提供的(android里由安卓sdk提供),具体可参看文末提供的官网链接里的例子。由于代码量较多就不贴了。

  • 还有个泛型的应用:

public static <X, Y> void processElements(
            Iterable<X> source,
            Predicate<X> tester,
            Function <X, Y> mapper,
            Consumer<Y> block) {
        for (X p : source) {
            if (tester.test(p)) {
                Y data = mapper.apply(p);
                block.accept(data);
            }
        }
    }
    
调用时:
processElements(
        new ArrayList<Person>(3),
        p -> p.getAge() <= 25,
        p -> p.getAge(),
        age -> System.out.println(age)
);

这个泛型要注意的是,它的类型由最先使用到它的那个地方来确定,比如Y类型,就是由最先使用到它的mapper返回getAge时来确定是int类型的,在block输入Y时就是个int类型,也就是在打印时age是个int类型。

  • 另外,顺带讲一下集合的聚合操作

java8对响应式编程的这一点支持是最值得称赞的,这一块需要另外讲一篇。我们还是先讲回lambda表达式。
其实这个跟lambda没有什么必然联系,聚合操作是java8为集合提供的一个方便的流式写法。只不过由于其中的参数都刚好是函数式接口,所以也可以用lambda来表示而已。它也可以不用lambda来表示,丝毫不影响这个聚合操作的优点发光发热。

new ArrayList<Person>()
      .stream()
      .filter(p -> p.getAge() >= 18 && p.getAge() <= 25)
      .map(p -> p.getAge())//相当于Function接口,map在这里是(函数)映射的意思
      .forEach(age -> System.out.println(age));

Lambda表达式变量的作用域

这一点的最后,还要提一下变量在lambda表达式中的作用域(这才是最实在的语法糖)

1.lambda函数体里可直接访问方法传递过来的参数,但此时相当于认为该参数是final型的,不允许再次赋值
2.lambda函数体里的this指的是函数外面最近一层类的实例,不是指函数所代表的接口实例

方法引用

方法引用是在lambda表达式中特定情形下,使用的特定表示语法。
具体限定为:当传过来的参数列表跟要调用的方法的参数能对应上,并且lambda函数体内只有一个方法调用语句时,如果符合以下四种情形之一,可以使用方法引用的语法(使用双冒号::)来表示。

1.静态方法

ContainingClass::staticMethodName

2.实例方法

containingObject::instanceMethodName

3.特定类型的实例方法

ContainingType::methodName

4.构造器的使用

ClassName::new

比如:
Arrays.sort(rosterAsArray, Person::compareByAge);
相当于:
Arrays.sort(rosterAsArray,
    (a, b) -> Person.compareByAge(a, b)
);
这是静态方法类型的使用,其它类型可参见官网相关章节。

需要注意的是,它只能在代替lambda表达式的时候使用。也就限定了只能在函数式接口上使用了,不能单独当作一句普通表达式来用的。比如,单独写这么一句:
Person::compareByAge;//这是不行的,哪怕那个方法是个不需要参数的也不行。

官方文档地址:
https://docs.oracle.com/javase/tutorial/java/javaOO/lambdaexpressions.html

相关文章:
http://blog.csdn.net/qq_28899635/article/details/53691986

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容

  • Java8 in action 没有共享的可变数据,将方法和函数即代码传递给其他方法的能力就是我们平常所说的函数式...
    铁牛很铁阅读 1,223评论 1 2
  • 简介 概念 Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主...
    刘涤生阅读 3,199评论 5 18
  • 在C++11中,我们还是会看到一些新元素。这些新鲜出炉的元素可能会带来一些习惯上的改变,不过权衡之下,可能这样的改...
    认真学计算机阅读 5,474评论 1 27
  • lambda表达式(又被成为“闭包”或“匿名方法”)方法引用和构造方法引用扩展的目标类型和类型推导接口中的默认方法...
    183207efd207阅读 1,476评论 0 5
  • 注:之前关于Java8的认知一直停留在知道有哪些修改和新的API上,对Lambda的认识也是仅仅限于对匿名内部类的...
    mualex阅读 2,821评论 1 4