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
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值