听说你还不会用Lambda表达式,那你还不点进来看?

本文介绍了Java8中新增的Lambda表达式特性,包括其引入背景、语法结构、功能接口等概念,以及如何使用Lambda表达式简化代码。同时,文章提供了多个实际应用场景的示例,如线程初始化、事件处理等。

1. 引言

在 Java 8 以前,若我们想要把某些功能传递给某些方法,总要去写匿名类。以前注册事件监听器的写法与下面的示例代码就很像:

manager.addScheduleListener(new ScheduleListener() {
 @Override
 public void onSchedule(ScheduleEvent e) { 
 // Event listener implementation goes here...
 }
});

这里我们添加了一些自定义代码到 Schedule 监听器中,需要先定义匿名内部类,然后传递一些功能到 onSchedule 方法中。

正是 Java 在作为参数传递普通方法或功能的限制,Java 8 增加了一个全新语言级别的功能,称为 Lambda 表达式

 

2. 为什么 Java 需要 Lambda 表达式

Java 是面向对象语言,除了原始数据类型之处,Java 中的所有内容都是一个对象。而在函数式语言中,我们只需要给函数分配变量,并将这个函数作为参数传递给其它函数就可实现特定的功能。JavaScript 就是功能编程语言的典范(闭包)。

Lambda 表达式的加入,使得 Java 拥有了函数式编程的能力。在其它语言中,Lambda 表达式的类型是一个函数;但在 Java 中,Lambda 表达式被表示为对象,因此它们必须绑定到被称为功能接口的特定对象类型。

3. Lambda 表达式简介

Lambda 表达式是一个匿名函数(对于 Java 而言并不很准确,但这里我们不纠结这个问题)。简单来说,这是一种没有声明的方法,即没有访问修饰符,返回值声明和名称。

在仅使用一次方法的地方特别有用,方法定义很短。它为我们节省了,如包含类声明和编写单独方法的工作。

Java 中的 Lambda 表达式通常使用语法是 (argument) -> (body),比如:

(arg1, arg2...) -> { body }
(type1 arg1, type2 arg2...) -> { body }

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

(int a, int b) -> { return a + b; }
() -> System.out.println("Hello World");
(String s) -> { System.out.println(s); }
() -> 42
() -> { return 3.1415 };

3.1 Lambda 表达式的结构

Lambda 表达式的结构:

  • Lambda 表达式可以具有零个,一个或多个参数。
  • 可以显式声明参数的类型,也可以由编译器自动从上下文推断参数的类型。例如 (int a) 与刚才相同 (a)。
  • 参数用小括号括起来,用逗号分隔。例如 (a, b) 或 (int a, int b) 或 (String a, int b, float c)。
  • 空括号用于表示一组空的参数。例如 () -> 42。
  • 当有且仅有一个参数时,如果不显式指明类型,则不必使用小括号。例如 a ->return a*a。
  • Lambda 表达式的正文可以包含零条,一条或多条语句。
  • 如果 Lambda 表达式的正文只有一条语句,则大括号可不用写,且表达式的返回值类型要与匿名函数的返回类型相同。
  • 如果 Lambda 表达式的正文有一条以上的语句必须包含在大括号(代码块)中,且表达式的返回值类型要与匿名函数的返回类型相同。

4. 方法引用

4.1 从 Lambda 表达式到双冒号操作符

使用 Lambda 表达式,我们已经看到代码可以变得非常简洁。

例如,要创建一个比较器,以下语法就足够了

Comparator c = (Person p1, Person p2) -> p1.getAge().compareTo(p2.getAge());

然后,使用类型推断:

Comparator c = (p1, p2) -> p1.getAge().compareTo(p2.getAge());

但是,我们可以使上面的代码更具表现力和可读性吗?我们来看一下:

Comparator c = Comparator.comparing(Person::getAge);

使用 :: 运算符作为 Lambda 调用特定方法的缩写,并且拥有更好的可读性。

4.2 使用方式

双冒号(::)操作符是 Java 中的方法引用。当们使用一个方法的引用时,目标引用放在 :: 之前,目标引用提供的方法名称放在 :: 之后,即 目标引用::方法。比如:

Person::getAge;

在 Person 类中定义的方法 getAge 的方法引用。

然后我们可以使用 Function 对象进行操作:

// 获取 getAge 方法的 Function 对象
Function<Person, Integer> getAge = Person::getAge;
// 传参数调用 getAge 方法
Integer age = getAge.apply(p);

我们引用 getAge,然后将其应用于正确的参数。

目标引用的参数类型是 Function<T,R>,T 表示传入类型,R 表示返回类型。比如,表达式 person -> person.getAge();,传入参数是 person,返回值是 person.getAge(),那么方法引用 Person::getAge 就对应着 Function<Person,Integer> 类型。

5. 什么是功能接口(Functional interface)

在 Java 中,功能接口(Functional interface)指只有一个抽象方法的接口。

java.lang.Runnable 是一个功能接口,在 Runnable 中只有一个方法的声明 void run()。我们使用匿名内部类实例化功能接口的对象,而使用 Lambda 表达式,可以简化写法。

每个 Lambda 表达式都可以隐式地分配给功能接口。例如,我们可以从 Lambda 表达式创建 Runnable 接口的引用,如下所示:

Runnable r = () -> System.out.println("hello world");

当我们不指定功能接口时,这种类型的转换会被编译器自动处理。例如:

new Thread(
 () -> System.out.println("hello world")
).start();

在上面的代码中,编译器会自动推断,Lambda 表达式可以从 Thread 类的构造函数签名(public Thread(Runnable r) { })转换为 Runnable 接口。

@FunctionalInterface 是在 Java 8 中添加的一个新注解,用于指示接口类型,声明接口为 Java 语言规范定义的功能接口。Java 8 还声明了 Lambda 表达式可以使用的功能接口的数量。当您注释的接口不是有效的功能接口时, @FunctionalInterface 会产生编译器级错误。

以下是自定义功能接口的示例:

package com.wuxianjiezh.demo.lambda;
@FunctionalInterface
public interface WorkerInterface {
 public void doSomeWork();
}

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

package com.wuxianjiezh.demo.lambda;
@FunctionalInterface
public interface WorkerInterface {
 public void doWork();
 public void doMoreWork();
}

错误:

Error:(3, 1) java: 意外的 @FunctionalInterface 注释
 com.wuxianjiezh.demo.lambda.WorkerInterface 不是函数接口
 在 接口 com.wuxianjiezh.demo.lambda.WorkerInterface 中找到多个非覆盖抽象方法

一旦定义了功能接口,我们就可以利用 Lambda 表达式调用。例如:

package com.wuxianjiezh.demo.lambda;
@FunctionalInterface
public interface WorkerInterface {
 public void doWork();
}
class WorkTest {
 public static void main(String[] args) {
 // 通过匿名内部类调用
 WorkerInterface work = new WorkerInterface() {
 @Override
 public void doWork() {
 System.out.println("通过匿名内部类调用");
 }
 };
 work.doWork();
 // 通过 Lambda 表达式调用
 // Lambda 表达式实际上是一个对象。
 // 我们可以将 Lambda 表达式赋值给一个变量,就可像其它对象一样调用。
 work = ()-> System.out.println("通过 Lambda 表达式调用");
 work.doWork();
 }
}

运行结果:

通过匿名内部类调用
通过 Lambda 表达式调用

6. Lambda 表达式的例子

6.1 线程初始化

线程可以初始化如下:

// Old way
new Thread(new Runnable() {
 @Override
 public void run() {
 System.out.println("Hello world");
 }
}).start();
// New way
new Thread(
 () -> System.out.println("Hello world")
).start();

我们在使用IDEA的时候,如果写出Old way的代码,IDEA会提示我们将其转换为Lambda表达式的形式,为IDEA点赞!

看完这些内容,你还会说自己不会用Lambda表达式吗

IDEA自动检测并提示转换为Lambda表达式形式

我们将光标移动到灰色代码区域(new Runnable这里),使用快捷键alt+Enter就可以实现自动转换了。

看完这些内容,你还会说自己不会用Lambda表达式吗

自动转换为Lambda表达式

6.2 事件处理

事件处理可以用 Java 8 使用 Lambda 表达式来完成。以下代码显示了将 ActionListener 添加到 UI 组件的新旧方式:

// Old way
button.addActionListener(new ActionListener() {
 @Override
 public void actionPerformed(ActionEvent e) {
 System.out.println("Hello world");
 }
});
// New way
button.addActionListener( (e) -> {
 System.out.println("Hello world");
});

6.3 遍例输出(方法引用)

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

// old way
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));
// 使用 :: 的 Lambda 表达式
list.forEach(System.out::println);

这里顺便补充一下Arrays.asList()方法。Arrays.asList()将数组转换为集合后,底层其实还是数组,《阿里巴巴》Java 开发使用手册对于这个方法有如下描述:

看完这些内容,你还会说自己不会用Lambda表达式吗

 

阿里巴巴Java开发手-Arrays.asList()方法

如何正确的将数组转换为ArrayList?可以像下面这样(参见:stackoverflow- https://dwz.cn/vcBkTiTW)

List list = new ArrayList<>(Arrays.asList("a", "b", "c"))

6.4 逻辑操作

输出通过逻辑判断的数据。

package com.wuxianjiezh.demo.lambda;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
public class Main {
 public static void main(String[] args) {
 List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7);
 System.out.print("输出所有数字:");
 evaluate(list, (n) -> true);
 System.out.print("不输出:");
 evaluate(list, (n) -> false);
 System.out.print("输出偶数:");
 evaluate(list, (n) -> n % 2 == 0);
 System.out.print("输出奇数:");
 evaluate(list, (n) -> n % 2 == 1);
 System.out.print("输出大于 5 的数字:");
 evaluate(list, (n) -> n > 5);
 }
 public static void evaluate(List<Integer> list, Predicate<Integer> predicate) {
 for (Integer n : list) {
 if (predicate.test(n)) {
 System.out.print(n + " ");
 }
 }
 System.out.println();
 }
}

运行结果:

输出所有数字:1 2 3 4 5 6 7
不输出:
输出偶数:2 4 6
输出奇数:1 3 5 7
输出大于 5 的数字:6 7

6.4 Stream API 示例

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

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

// old way
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7);
for(Integer n : list) {
 int x = n * n;
 System.out.println(x);
}
// new way
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7);
list.stream().map((x) -> x*x).forEach(System.out::println);

下面的示例中,我们给定一个列表,然后求列表中每个元素的平方和。这个例子中,我们使用了 reduce() 方法,这个方法的主要作用是把 Stream 元素组合起来。

// old way
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7);
int sum = 0;
for(Integer n : list) {
 int x = n * n;
 sum = sum + x;
}
System.out.println(sum);
// new way
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7);
int sum = list.stream().map(x -> x*x).reduce((x,y) -> x + y).get();
System.out.println(sum);

7. Lambda 表达式和匿名类之间的区别

  • this 关键字。对于匿名类 this 关键字解析为匿名类,而对于 Lambda 表达式,this 关键字解析为包含写入 Lambda 的类。
  • 编译方式。Java 编译器编译 Lambda 表达式时,会将其转换为类的私有方法,再进行动态绑定。
excludes = {'什么', '一个', '我们', '你们', '如今', '说道', '知道', '起来', '这里','奶奶', '姑娘', '出来', '众人', '那里', '自己', '他们', '一面', '只见', '怎么','老太太', '两个', '没有', '不是', '不知', '这个', '听见', '这样', '来', '咱们','太太', '告诉', '就是', '东西', '回来', '只是', '大家', '只得', '丫头','姐姐','不用', '过来', '心里', '如此', '今日', '这些', '不敢', '出去', '所以', '不过', '的话', '不好', '一时', '不能', '银子', '几个', '答应', '二人', '还有', '只管', '这么', '说话', '一回', '那边', '这话', '外头', '打发', '自然', '今儿', '罢了', '屋里', '那些', '听说', '如何', '问道', '看见','二爷','小丫头','人家','妹妹','老爷'} txt = open("D:\python work\ReadHouse.txt", "r", encoding='utf-8').read() words = txt.split() counts = {} for word in words: if len(word) == 1: continue elif word == "宝玉" or word == "宝二爷" : rword = "贾宝玉" elif word == "凤姐" or word == "凤辣子" or word == "凤姐儿" : rword = "王熙凤" elif word == "老祖宗" or word == "老太君": rword = "贾母" elif word == "林姑娘" or word == "黛玉" or word == "林妹妹" or word == "林丫头": rword = "林黛玉" elif word == "宝姑娘" or word == "宝丫头" or word == "蘅芜君" or word == "宝姐姐" or word == "宝钗": rword = "薛宝钗" elif word == "湘云": rword = "史湘云" elif word == "花珍珠" or word == "花大姑娘": rword = "袭人" else : rword = word counts[rword] = counts.get(rword,0) + 1 for word in excludes: del counts[word] items = list(counts.items()) items.sort(key=lambda x:x[1], reverse=True) for i in range(20): word, count = items[i] print ("{0:{2}<10}{1:{2}>5}".format(word, count,chr(12288))) 分析并修改此段代码
最新发布
06-29
<think>嗯,用户问的是,如果一个接口有多个抽象方法,是否还能使用Lambda表达式。我记得Java中的Lambda表达式通常用于函数接口,也就是只有一个抽象方法的接口。那如果有多个抽象方法的话,是不是就不能用了呢?不过,可能存在一些特殊情况或者新的特性,我需要仔细想想。 首先,函数接口的定义是只有一个抽象方法的接口,这个我记得没错。所以如果接口有多个抽象方法,那它就不是函数接口,自然不能用Lambda表达式实现。但用户的问题可能需要更详细的解释,比如是否有例外情况,或者Java版本的变化。 然后,可能要考虑接口中的默认方法和静态方法。虽然这些方法可以有多个,但它们不影响抽象方法的数量。例如,一个接口如果有两个抽象方法,即使有很多默认方法,它仍然不是函数接口,所以Lambda不能应用。 另外,是否存在其他情况?比如,如果接口继承另一个接口,但总的抽象方法数量超过一个,这时候也不行。或者,用户可能误解了默认方法和抽象方法的区别,需要澄清。 还需要确认是否在Java 8之后有变化。比如,Java 8引入了函数接口,但后续版本有没有允许有多个抽象方法的接口使用Lambda?根据我的知识,应该没有,仍然严格要求只能有一个抽象方法。 可能的例子:用户定义了一个接口,里面有两个抽象方法,然后尝试用Lambda实现,这时候编译器会报错。这时候应该建议用户检查接口中的抽象方法数量,或者考虑拆分成多个接口。 总结的话,应该明确说明,只有函数接口才能用Lambda,而函数接口必须只有一个抽象方法,所以如果有多个就不能用。但需要举例子说明,可能还要提到@FunctionalInterface注解的作用,帮助用户检查接口是否符合条件。</think>好的,我们逐步分析这个问题: 1. **Lambda表达式的前提条件** - JavaLambda表达式只能用于实现**函数接口**(Functional Interface) - 函数接口的定义:有且仅有一个**抽象方法**的接口 2. **多抽象方法接口的后果** - 如果接口包含多个抽象方法,例如: ```java interface MyInterface { void method1(); void method2(); // 第二个抽象方法 } ``` - 该接口将**不再是函数接口** - 尝试使用Lambda表达式编译器会报错: ```java MyInterface obj = () -> {}; // 错误:不兼容的类型 ``` 3. **特殊情况的验证** - **默认方法**和**静态方法**不影响: ```java interface MyInterface { void abstractMethod(); default void defaultMethod() {} // 不计入抽象方法数量 static void staticMethod() {} // 不计入抽象方法数量 } ``` - 上述接口仍可正常使用Lambda表达式 - **继承的抽象方法**会累计: ```java interface Parent { void parentMethod(); } interface Child extends Parent { void childMethod(); } // 总共有2个抽象方法,无法使用Lambda ``` 4. **解决方案建议** - 使用`@FunctionalInterface`注解强制校验: ```java @FunctionalInterface // 如果存在多个抽象方法,编译时会报错 interface ValidInterface { void singleMethod(); } ``` - 重构设计模式: - 策略模式:拆分不同行为到多个函数接口 - 建造者模式:通过方法链处理多个操作 **结论**:当接口包含多个抽象方法时,无法直接使用Lambda表达式实现。必须保证接口符合函数接口的定义(单一抽象方法)才能使用Lambda表达式。这是Java语言规范对Lambda使用场景的明确限制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值