Java8新特征之Lambda

Wesley13
• 阅读 696

Java8新特征之Lambda

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

注意:Lambda中必须有个接口

3、Lambda的形式

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

3.1、 无参带返回值

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

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

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

以上的Lambda等价于:

Model2 md2 = () -> (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表达式,其中省略掉了每一个参数的类型,编译器自动推断。多条语句时实现体的{}不能省。

最新技术学习资料(^_^) → lezijie007(程序员暗号:513cto13)

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作为参数
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
Java8新特征之Lambda
中的抽象方法只有accept一个:

Java8新特征之Lambda

​ 通过在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

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Wesley13 Wesley13
2年前
Java爬虫之JSoup使用教程
title:Java爬虫之JSoup使用教程date:201812248:00:000800update:201812248:00:000800author:mecover:https://imgblog.csdnimg.cn/20181224144920712(https://www.oschin
Wesley13 Wesley13
2年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
2年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
2年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
2年前
Docker 部署SpringBoot项目不香吗?
  公众号改版后文章乱序推荐,希望你可以点击上方“Java进阶架构师”,点击右上角,将我们设为★“星标”!这样才不会错过每日进阶架构文章呀。  !(http://dingyue.ws.126.net/2020/0920/b00fbfc7j00qgy5xy002kd200qo00hsg00it00cj.jpg)  2
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这