学java怎么能不会用Lambda表达式呢

学习目标:

简单学会Lambda表达式

学习内容:

1、 背景 2、 简介 3、 结构 4、 功能接口 5、 使用案例

学习时间:

2022年1月7日

学习产出:

1、 技术笔记 1 遍 2、CSDN 技术博客 1 篇

背景

java是一流的面向对象语言,除了部分简单的数据类型,java中的一切都是对象,即使数组也是一种对象,每个类创建的实例也是对象。在java中定义的函数或方法不可能完全独立,也不能将方法作为参数或返回一个方法给实例。

在java8以前,若我们想要把某些功能传递给某个方法,总要去写匿名类。

示例:

list.sort(new Comparator<User>() { 
    @Override 
    public int compare(User o1, User o2) { 
        return o1.getId()-o2.getId(); 
    } 
}

​ 在上面的例子里,为了对集合集合进行排序,我们为 Comparator 接口创建了一个它的匿名内部类对象,重写接口中的方法,来实现排序功能.

​ 简而言之,在 Java 里将普通的方法或函数像参数一样传值并不简单,为此, Java 8 增加了一个语言级的新特性,名为 Lambda 表达式.

简介

​ Lambda 表达式是一个匿名函数,我们可以把 lambda 表达式理解为一段可以传递的代码(将代码段像数据一样传递)。使用它可以写出更简洁, 更灵活的代码。作为一种更紧凑的代码风格,使 java 语言的表达式能力得到的提升。

​ Lambda 表达式的本质只是一个"语法糖",由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能。

Java 中的 Lambda 表达式通常使用 **(argument) -> {body}**语法书写,例如:

左侧:lambda 表达式的参数列表 
右侧:lambda 表达式中需要执行的功能,即 lambda 体 
(arg1, arg2...) -> { body } 
(type1 arg1, type2 arg2...) -> { body }

以下是一些Lambda表达式的例子:

无参数,无返回值,lambda 体中只有一行代码时,{}可以忽略 
() -> System.out.println("Hello World"); 
无参数,有返回值 
() -> { return 3.1415 }; 
有参数,无返回值 
(String s) -> { System.out.println(s); } 
有一个参数,无返回值 
s -> { System.out.println(s); } 
有多个参数,有返回值 
(int a, int b) -> { return a + b; } 
有多个参数,表达式参数类型可以不写,jvm 可以根据上下文进行类型推断 (a, b) -> { return a - b; }

Lambda表达式的结构

1、Lambda 表达式可以具有零个,一个或多个参数。

2、可以显式声明参数的类型,也可以由编译器自动从上下文推断参数的类型。

​ 例如 (int a,int b)与 (a,b)相同。

3、参数用小括号括起来,用逗号分隔。例如 (a, b) 或 (int a, int b) 或 (String a, int b, float c)。

4、空括号用于表示一组空的参数。

​ 例如 () -> 42。

5、当有且仅有一个参数时,如果不显式指明类型,则不必使用小括号。

​ 例如 a -> return a*a。

6、Lambda 表达式的正文可以包含零条,一条或多条语句。

7、如果 Lambda 表达式的正文只有一条语句,则大括号可不用写,且表达式 的返回值类型要与匿名函数的返回类型相同。

8、如果 Lambda 表达式的正文有一条以上的语句必须包含在大括号(代码块)中,且表达式的返回值类型要与匿名函数的返回类型相同。

功能接口

功能接口是 java 8 中的新增功能,它们只允许一个抽象方法。这些接口也称为单抽象方法接口。这些也可以使用 Lambda 表达式,方法引用和构造函数引用来表示。Java 8 也引入了一个注释,即@FunctionalInterface,当你注释的接口违反了 Functional Interface 的契约时,它可以用于编译器级错误。

自定义功能接口:

@FunctionalInterface 
public interface WorkerInterface { 
    public void doSomeWork(); 
}

正如其定义所述,功能接口只能有一个抽象方法。如果我们尝试在其中添加一个抽象方法,则会抛出编译时错误。例如:

@FunctionalInterface 
public interface WorkerInterface { 
	//功能接口中有两个方法是错误的
	public void doWork(); 
	public void doMoreWork(); 
}

使用案例

@FunctionalInterface 
public interface Shape { 
	void area(int a,int b); 
}
Shape d = (a, b)->{System.out.println(a+b);}; 
	  d.area(1, 2);

使用案例

线程初始化

//匿名内部类方式 
new Thread(new Runnable() { 
    @Override 
    public void run() { 
        System.out.println("Hello world"); 
    } 
}).start(); 
// lambda 表达式方式 
new Thread( () -> System.out.println("Hello world") ).start();

事件处理

// 匿名内部类方式
button.addActionListener(new ActionListener() {
    @Override 
    public void actionPerformed(ActionEvent e) { 
        System.out.println("Hello world"); 
    } 
}); 
// lambda 表达式方式 
button.addActionListener( (e) -> { 
    System.out.println("Hello world"); 
});

遍历输出(方法引用)

输出给定数组的所有元素的简单代码。请注意,还有一种使用 Lambda 表达式的方式。

//普通方式 
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7); 
for (Integer n : list) { 
    System.out.println(n); 
}
// 使用 -> 的 Lambda 表达式 
list.forEach(n -> System.out.println(n));

Stream API 示例

java.util.stream.Stream 接口 和 Lambda 表达式一样,都是 Java 8 新引入的。所有 Stream 的操作必须以 Lambda 表达式为参数。Stream 接口中带有 大量有用的方法,比如 map() 的作用就是将 input Stream 的每个元素,映射成 output Stream 的另外一个元素。

下面的例子,我们将 Lambda 表达式 x -> x*x 传递给 map() 方法,将其应 用于流的所有元素。之后,我们使用 forEach 打印列表的所有元素。

//原始方式 
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7); 
for(Integer n : list) { 
    int x = n * n; 
    System.out.println(x); 
}
//lambda 方式 
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7); 
list.stream().forEach((n)-> System.out.println(n));
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值