148. Java Lambda 表达式 - 捕获局部变量

148. Java Lambda 表达式 - 捕获局部变量

一旦你习惯了 Lambda 表达式的使用,它们会变得非常自然。LambdaCollections FrameworkStream API 以及 JDK 中的许多其他部分无缝集成。从 Java SE 8 开始,Lambda 表达式几乎无处不在,它们成为了 Java 编程中的一个重要特性。

但是,使用 Lambda 表达式时有一些限制,可能会导致你遇到编译时错误,了解这些限制非常重要。

让我们看一下以下的代码示例:

java 复制代码
int calculateTotalPrice(List<Product> products) {
    int totalPrice = 0;  // 局部变量
    Consumer<Product> consumer = product -> totalPrice += product.getPrice();  // Lambda 表达式

    for (Product product : products) {
        consumer.accept(product);  // 调用 consumer 的 accept 方法
    }

    return totalPrice;
}

看起来这段代码是没有问题的,但实际上,编译器会给出如下错误:

"lambda 表达式中使用的变量应为 final 或实际上是 final"

原因:

Lambda 表达式无法修改它所捕获的外部变量。Lambda 表达式可以读取外部变量的值,但前提是这些变量必须是 final 的或者实际上是 final 的。捕获外部变量意味着 Lambda 只能访问并使用它们的值,而不能修改它们。

何为"final"变量?

Java 中,final 变量是不可变的,它的值一旦被赋值就不能再更改。在 Lambda 表达式中,final 变量实际上是常量,可以直接读取。但 Lambda 表达式不能修改它们,因为它们在 Lambda 内部的作用域中是只读的。

为什么必须是 final 或 实际上是 final

Java 中的 Lambda 表达式不能修改它们所捕获的外部变量,这是为了保证并发安全性和避免不必要的副作用。如果允许修改外部变量,会引发线程安全问题或难以预料的行为。

Java SE 8 引入了"实际上是 final "的概念。即使你没有显式声明变量为 final,编译器也会为你推断出它是否是 final。如果一个变量在 Lambda 表达式中被引用,但没有被修改,那么编译器会自动将其视为 final。因此,你不一定要显式地加上 final 关键字,编译器会根据使用情况自动处理。

例子:修复代码中的问题

如果我们修改代码中的 totalPrice 变量,使它在 Lambda 表达式中变为"实际上是 final ",编译器就不会报错了。为了避免错误,可以使用 AtomicInteger 来处理这种情况,因为 AtomicInteger 是可变的,可以在 Lambda 中进行修改。修改后的代码如下:

java 复制代码
import java.util.concurrent.atomic.AtomicInteger;

int calculateTotalPrice(List<Product> products) {
    AtomicInteger totalPrice = new AtomicInteger(0);  // 使用 AtomicInteger
    Consumer<Product> consumer = product -> totalPrice.addAndGet(product.getPrice());  // 通过 addAndGet 修改值

    for (Product product : products) {
        consumer.accept(product);
    }

    return totalPrice.get();  // 返回最终的价格
}

这里使用 AtomicInteger 来代替原始的 int 类型,因为 AtomicInteger 提供了线程安全的可变值,可以安全地在 Lambda 表达式中进行修改。

总结:

  • Lambda 表达式只能捕获final或"实际上是final"的局部变量。 也就是说,Lambda 只能读取外部变量的值,不能修改它们。
  • "实际上是 final"Java SE 8 中引入的新概念,意味着编译器可以自动推断哪些变量是不可修改的,从而避免你显式声明 final
  • 如果需要修改外部变量的值,可以使用线程安全的类,如 AtomicInteger 来替代普通变量。

通过理解这些限制和解决方案,可以更好地使用 Lambda 表达式,并避免常见的编译错误。

相关推荐
永不停歇的蜗牛17 小时前
Maven的POM文件相关标签作用
服务器·前端·maven
计算机毕设小月哥17 小时前
【Hadoop+Spark+python毕设】中风患者数据可视化分析系统、计算机毕业设计、包括数据爬取、Spark、数据分析、数据可视化、Hadoop
后端·python·mysql
q***441517 小时前
Spring Security 新版本配置
java·后端·spring
芳草萋萋鹦鹉洲哦17 小时前
【vue/js】文字超长悬停显示的几种方式
前端·javascript·vue.js
计算机毕设匠心工作室17 小时前
【python大数据毕设实战】强迫症特征与影响因素数据分析系统、Hadoop、计算机毕业设计、包括数据爬取、数据分析、数据可视化、机器学习、实战教学
后端·python·mysql
o***741717 小时前
Springboot中SLF4J详解
java·spring boot·后端
HIT_Weston18 小时前
47、【Ubuntu】【Gitlab】拉出内网 Web 服务:Nginx 事件驱动分析(一)
前端·ubuntu·gitlab
雨中散步撒哈拉18 小时前
18、做中学 | 初升高 | 考场一 | 面向过程-家庭收支记账软件
开发语言·后端·golang
开发者小天18 小时前
React中的 闭包陷阱
前端·javascript·react.js
翔云 OCR API18 小时前
承兑汇票识别接口技术解析-开发者接口
开发语言·前端·数据库·人工智能·ocr