函数式接口和Lambda表达式深入理解


我上一篇文章介绍了函数式接口和Lambda表达式,以及Java解决所谓的闭包。
这次深入一下。

0x00 函数式接口

前面讲了一下函数式接口,不过可能只是讲了个大概,大致讲了一下什么是函数式接口

  • 函数式接口就是:一个interface,里面只有一个抽象方法,其他什么都没有。
  • FunctionalInterface注解标注一个函数式接口,不能标注方法枚举属性这些。
  • 如果接口被标注了@FunctionalInterface,这个类就必须符合函数式接口的规范
  • 即使一个接口没有标注@FunctionalInterface,如果这个接口满足函数式接口规则,依旧被当作函数式接口。

这次我们来用代码来深入了解函数式接口

Demo01

如上图,只包含一个抽象方法是最普通的函数式接口

两个抽象方法,报错

再看,当接口有两个抽象方法的时候,就不在是函数式接口了,使用@FunctionalInterface标注编译时会报错

特例-01

奇怪的是这里有3个抽象方法,为什么不报错?
我们知道toStringequals方法是Object的方法,Java基础告诉我们,Object是所有类的默认父类,也就是说任何对象都会包含Object里面的方法,即使是函数式接口的实现,也会有Object的默认方法,所以:重写Object中的方法,不会计入接口方法中,除了final不能重写的,Object中所能重写的方法,写到接口中,不会影响函数式接口的特性

特例-02

Java8 允许接口中含有非抽象方法,这种在接口中使用default修饰的非抽象方法称为默认方法,默认方法也不会影响函数式接口的特性。我们依然可以认为DemoConsumer是一个函数式接口。

0x01 Lambda表达式深入

Lambda表达式的形式如下

(param1, param2, param3, param4…)->{ doing……};

由此引申出多种写法:

//1.
() -> System.out.println("Hello Lambda");
//2.
number1 -> int a = number1 * 2;
//3.
(number1, number2) -> int a = number1 + number2;
//4.
(number1, number2) -> {
 int a = number1 + number2;
 System.out.println(a);
}

下面通过重构一段代码,来深入了解一下Lambda表达式

public class FunctionalInterfaceTest {
    public static void main(String[] args) {
        List<String> demoList = Arrays.asList( "Zing", "阿三", "小明", "小红", "赵日天");
       rollCall(demoList);
    }
    public static void rollCall(List<? extends String> list){
        for(String name : list){
            if(name.startsWith("小")){
                System.out.println(name);
            }
        }
    }
}

如果希望筛选的条件能自由定义,而不是name.startsWith("小")写死,并且希望找到人名后,不是简单的System.out.println(name);,而是能做一些其他的事情。
继续重构:

/**
 * 函数式接口
 * @param <T>
 */
@FunctionalInterface
interface Checker<T extends String>{
    boolean check(T t);
}

@FunctionalInterface
interface Out<T>{
    void achievement(T t);
}

public class FunctionalInterfaceTest {
    /**
     * 点名
     */
    @Test
    public void testLambda() {
        List<String> demoList = Arrays.asList("小明", "Zing", "阿三", "小红", "赵日天");
        rollCall(demoList,
                name-> name.startsWith("Z"),
                name->{
                    String rate = name + "是单身狗!";
                    System.out.println(rate);
                });
    }

    /**
     * 点名逻辑
     * @param list
     * @param checker
     */
    public void rollCall(List<? extends String> list, Checker checker,Out out){
        for(String name : list){
            if(checker.check(name)){
                out.achievement(name);
            }
        }
    }
}

运行结果

一不小心暴露了什么。哈哈哈

通过上面的重构,很明显,这么写也是合法的

        Checker checker =  name-> name.startsWith("Z"),
        Out estimator = name->{
            String rate = name + "是单身狗!";
            System.out.println(rate);
        };

由此可以知道,Lambda和函数式接口是等价的。

0x02 补充

  • 类型
    有人会很奇怪,为什么Checker checker = name-> name.startsWith("Z")这样写的时候,name会被当成String 类型?

这是Java的类型推断,大致逻辑是编译器知道函数式接口方法的输入参数类型,所以无论前面的参数是什么名字,都会被当成方法所需要的参数类型。

  • 简单缩写
    还有一个奇怪的地方name->name.startsWith("Z")为什么这样写也可以?
    为什么不是写成``name->{ return name.startsWith("Z");}` 。
    很明显,后面的写法是没有错的,

但是Idea会有一个虚线,说明不需要写return


展开看说明

当只需要执行一条语句的时候,lambda支持这种简洁返回。所以为什么拒绝呢?

  • 外部参数
    Lambda表达式是不能操作外部对象的,因为Lambda 实质上是接口的子对象,只能访问静态资源和本身的内部变量。

报错!

编译器会要求将外部变量使用final修饰。

  • 和方法引用结合
    方法引用Method References是Java8配合Lambda一起做出的新特性,当Lambda表达式里面只执行已知的方法的时候,可以使用方法引用来写出跟简洁易读的代码
List<String> demoList = Arrays.asList("小明", "Zing", "阿三", "小红", "赵日天");
demoList.forEach(System.out::println);

看到这里想必心里不禁想说,我擦,好简洁!
官方给出了4种方法引用

Kinds of Method References

| Kind | Example|
| ---- |----|
|Reference to a static method|ContainingClass::staticMethodName|
|Reference to an instance method of a particular object|containingObject::instanceMethodName|
|Reference to an instance method of an arbitrary object of a particular type|ContainingType::methodName|
|Reference to a constructor| ClassName::new|
来源:
http://docs.oracle.com/javase/tutorial/java/javaOO/methodreferences.html

我想我就不用翻译了吧,出门百度翻译😉


love&peace
FS全栈计划目录:https://micorochio.github.io/fs-plan/

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

推荐阅读更多精彩内容