Variable used in lambda expression should be final or effectively final

问题描述

在使用java8lambda表达式的时候,有时候会遇到这样的编译报错
在这里插入图片描述
这句话的意思是,lambda表达式中使用的变量应该是final或者是有效的final。在Java8之前,匿名类中如果要访问局部变量的话,那个局部变量必须显式的声明为final。而在Java8之后,在匿名类活lambda表达式中访问的局部变量,如果不是final类型的话,编译器自动加上final修饰符。

为什么lambda表达式或者匿名内部类不能访问非final的局部变量?

在lambda表达式中对变量的操作都是基于原变量的副本,不会影响到原变量的值。假定没有要求lambda表达式外部变量为final修饰,那么开发者会误以为外部变量的值能够在lambda表达式中被改变,而这实际是不可能的,所以要求外部变量为final是在编译期以强制手段确保用户不会在lambda表达式中做修改原变量的操作;
其实这就要说到JVM内存模型和线程了,因为实例变量存在堆中,而局部变量是在栈上分配,lambda表达式(匿名内部类)会在另一个线程中执行。如果在线程中要直接访问一个局部变量,可能线程执行时该局部变量已经被销毁了,而final类型的局部变量在lambda表达式(匿名类)中其实是局部变量的一个拷贝。

解决方案

方案一

使用常量且只能存放一个元素的数组

public class CustomInterceptor implements Ratify {

    @Override
    public Result deal(Chain chain) {
        Request request = chain.request();
        System.out.println("CustomInterceptor=====>" + request.toString());
        final Result[] result = {new Result(true, "同意请假")};
        Optional.ofNullable(request.getReason()).ifPresent(reason -> {
            if ("事假".equals(reason)) {
                Request newRequest = new Request.Builder()
                        .newRequest(request)
                        .customInfo(request.getName() + "请的是事假,而且很着急,请领导重视一下")
                        .build();
                System.out.println("CustomInterceptor=====>转发请求");
                result[0] = chain.proceed(newRequest);
            }
        });
        return result[0];
    }

}

方案二

由于在lambda表达式中对变量的操作都是基于原变量的副本,这才导致不能改变原变量的值。所以,我们可以推出可以使用原子引用AutomicReference来提供一种对原子变量的对象的引用机制。

public class CustomInterceptor implements Ratify {

    @Override
    public Result deal(Chain chain) {
        Request request = chain.request();
        System.out.println("CustomInterceptor=====>" + request.toString());
        AtomicReference<Result> result = new AtomicReference<>(new Result(true, "同意请假"));
        Optional.ofNullable(request.getReason()).ifPresent(reason -> {
            if ("事假".equals(reason)) {
                Request newRequest = new Request.Builder()
                        .newRequest(request)
                        .customInfo(request.getName() + "请的是事假,而且很着急,请领导重视一下")
                        .build();
                System.out.println("CustomInterceptor=====>转发请求");
                result.set(chain.proceed(newRequest));
            }
        });
        return result.get();
    }

}
  • 14
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
"variable used in lambda expression should be final or effectively final"这个报错是由于在Lambda表达式中使用的变量需要被声明为final或者有效地finalLambda表达式是由匿名内部类演变而来的,而匿名内部类中使用的变量也需要是final类型。这是因为Lambda表达式可以访问外部作用域中的变量,但是不能修改这些变量的值,以确保程序的正确性和可靠性。因此,编译器会对没有被声明为final或有效final的变量进行警告。这个限制的目的是为了避免在Lambda表达式中对外部变量进行修改而引发的潜在问题。通过将变量声明为final或有效final,可以确保在Lambda表达式中使用这些变量时的一致性和可预测性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [编译器说 Lambda 表达式中的变量必须是 final 的,我偏不信](https://download.csdn.net/download/weixin_38651286/13755089)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [【Java 錯誤】Variable used in lambda expression should be final or effectively final](https://blog.csdn.net/qq_45802080/article/details/125451124)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值