乐字节-Java8新特性-Lambda表达式

上一篇文章我们了解了Java8新特性-接口默认方法,接下来我们聊一聊Java8新特性之Lambda表达式。

Lambda表达式(也称为闭包),它允许我们将函数当成参数传递给某个方法,或者把代码本身当作数据处理。很多语言(Groovy、Scala等)从设计之初就支持Lambda表达式。但是java中使用的是 匿名内部类代替

最后借助强大的社区力量,找了一个折中的Lambda实现方案,可以实现简洁而紧凑的语言结构。

1、匿名内部类到Lambda的演化

匿名内部类,即一个没有名字的,存在于一个类或方法内部的类。当我们需要用某个类且只需要用一次,创建和使用和二为一时,我们可以选择匿名内部类,省掉我们定义类的步骤。

匿名内部类会隐士的继承一个类或实现一个接口,或者说匿名内部类是一个继承了该类或者实现了该接口的子类匿名对象。下面看一个匿名内部类的例子:

测试类中调用方法

package com.lotbyte.main;

/*

    定义和使用匿名内部类

*/

public class NoNameClass {

    public static void main(String[] args) {

        Model m = new Model(){

            @Override

            public void func() {

                System.out.println("方法的实现");

            }

        };

        m.func();

    }

}

// 需要被实现的接口

interface Model{

    void func();

}

2、Lambda快速使用

从某种意义上来说,Lambda表达式可以看作是匿名内部类对象的简写形式。最简单的Lambda表达式可以由 用逗号分隔的参数列表、->符号和语句块组成。

注意:此时匿名内部类只能实现接口,不能是继承抽象类

例如将上面的例子做一个简化,使用Lambda的形式如下:

public class NonameClassForLambda {

    public static void main(String[] args) {

        // Lambda方式简写,方法实现可以很简单

        Model1 md = ()-> System.out.println("hello");

        md.func();


        // 也可以是比较复杂的操作

        md = () -> {

            for (int i = 1; i <=5; i++) {

                System.out.println(i);

            }

        };

        md.func();

    }

}

// 接口

interface Model1{

    void func();

}


以上是一个简单的Lambda的书写形式,()中是形参列表,没有则为空括号, ->为语法格式,之后则为方法的实现(一条语句可以直接书写,当有多条语句时,需要使用{}进行包裹)。从这可以看出在接口中必须只能存在一个抽象方法。

注意:Lambda中必须有个接口


3、Lambda的形式

使用Lambda时,实现方法可以有参数,也可以有返回值,如果没指定参数类型,则由编译器自行推断得出。

3.1、 无参带返回值

生成[1,10]之间的任意整数

说明:Lambda的改写需要有对应的抽象方法,当没有参数时需要使用()占位,当表达式只有一行代码时,可以省略return和{}

interface Model2{

    int func();

}

Model2 md2 = () -> {return (int)(Math.random()*10+1)};

以上的Lambda等价于:

Model2md2=()->(int)(Math.random()*10+1);

3.2 、带参带返回值

返回一个对数字描述的字符串

interface Model3{

    String func(int a);

}

Model3 md3 = (int a) -> {

    return "This is a number " + a;

};

说明:形参写在()内即可,参数的类型可以省略,此时将由编译器自行推断得出,同时还可以省略()

md3=a->"This is a number "+a;

省略了参数类型,小括号,同时连带实现体的括号和return都省了。

3.3 、带多个参数

根据输入的运算符计算两个数的运算,并返回结果:

interface Model4{

    String func(int a, int b, String oper);

}

Model4 md4 = (a, b, s) -> {

      String res = "";

      if("+".equals(s)){

            res = ( a+b ) + "";

      }else if("-".equals(s)){

            res = ( a-b ) + "";

      }else if("*".equals(s)){

            res = ( a*b ) + "";

      }else if("/".equals(s)){

            res = ( a/b ) + ""; // 暂不考虑除0的情况

      }else{

            res =  "操作有失误";

      }

      return res;

};

System.out.println(md4.func(1,1,"+"));

以上例子为多个参数的Lambda表达式,其中省略掉了每一个参数的类型,编译器自动推断。多条语句时实现体的{}不能省。

4、Lambda作为参数

在jdk8之前,接口可以作为方法参数传入,执行时必须提供接口实现类的实例。从java8开始,Lambda可以作为接口方法实现,当作参数传入,无论从形式上还是实际上都省去了对象的创建。使代码更加的紧凑简单高效。

使用Lambda表达式需要有以下几步:

​ 1、定义接口,抽象方法的模板;

​ 2、在某方法中需要接口作为参数;

​ 3、调用方法时需要将抽象方法实现(此时我们使用Lambda表达式)并传入即可。


4.1、定义接口

在接口中,必须有且仅有一个抽象方法,以确定Lambda模板

// 无参无返回值的方法

interface LambdaInterface1{

    void printString();

}

// 带参无返回值的方法

interface  LambdaInterface2{

    void printString(String str);

}

4.2、定义方法接收参数

在某方法中需要使用接口作为参数

// 无参

public static void testLambda(LambdaInterface1 lam1){

    lam1.printString();

}

// 带参

public static void testLambda2(String s,LambdaInterface2 lam2){

    lam2.printString(s);

}

4.3、Lambda实现

使用方法时需要用Lambda将抽象方法实现

使用方法时需要用Lambda将抽象方法实现

// 无参Lambda作为参数

testLambda(()->{

    System.out.println("可以简单,可以复杂");

});

// 带参Lambda作为参数

testLambdaParam("hello",(a)->{

    System.out.println(a);

});

通过以上三步,能够完整地展示Lambda从和演变而来。此后在使用时,jdk中已经提供很多场景了,即前两部已经完成,我们更多的是实现第三步即可。

5、forEach展示Lambda

例如以ArrayList的遍历为例子,分析Lambda的使用方式。

public static void main(String[] args) {

    List<String> strs = new ArrayList<String>(){

        {

            add("aaa");

            add("bbb");

            add("ccc");

        }

    };

    strs.forEach((str)-> System.out.println(str));

}

下面看看forEach的源码,定义中使用了接口Consumer作为参数,并调用了其方法:


Consumer中的抽象方法只有accept一个


通过在forEach方法中调用Consumer的accept方法,并将每一个元素作为参数传入,使得accept方法可以对每一个元素进行操作,当我们使用Lambda实现accept时就变成了我们自己对每一个元素的处理了。我们只负责处理即可。

6、Lambda中使用变量

​ 在Lambda中可以定义自己的局部变量,也可以使用外层方法的局部变量,还可以使用属性。这一点也不难理解,既然是一个方法的实现,只写了一个代码块,那么使用本身所属方法的局部变量和类的属性也并不过分。

public static void main(String[] args) {

    List<String> strs = new ArrayList<String>(){

        {

            add("aaa");

            add("bbb");

            add("ccc");

        }

    };

    int j = 1;

    strs.forEach((str)->{

        int i = 0;

        System.out.println(str + "  " + i + "  " + j);

    });

}

注意:此时外部局部变量将自动变为final

7、Lambda作为方法返回值

例子:返回判断字符串是否为空

public class Demo004_2 {

    public static void main(String[] args) {

        System.out.println(testLambda().isEmpty("string"));

    }

    // 判断字符串是否为空

    public static AssertEmpty testLambda(){

        return (n)-> null==n||n.trim().isEmpty(n);

    }

}

interface AssertEmpty{

    boolean isEmpty(String str);

}

今天关于Java8新特性-Lambda表达式就讲到这里了,接下来我会继续讲述Java8新特性之函数式接口。敬请继续关注!欢迎留言评论。

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

推荐阅读更多精彩内容

  • Java 8自Java 5(发行于2004)以来最具革命性的版本。Java 8 为Java语言、编译器、类库、开发...
    谁在烽烟彼岸阅读 888评论 0 4
  • Java 8自Java 5(发行于2004)以来最具革命性的版本。Java 8 为Java语言、编译器、类库、开发...
    huoyl0410阅读 625评论 1 2
  • java8新特性 原创者:文思 一、特性简介 速度更快 代码更少,增加了Lambda 强大的Stream API ...
    文思li阅读 3,045评论 1 1
  • lambda表达式(又被成为“闭包”或“匿名方法”)方法引用和构造方法引用扩展的目标类型和类型推导接口中的默认方法...
    183207efd207阅读 1,477评论 0 5
  • 2018年5月11日早晨,朋友在微信群里面发出了如下消息。 果不其然,随后币值大幅下跌。第二天也就是5...
    只想飞的猪阅读 567评论 0 1