Variable used in lambda expression should be final or effectively final

问题描述

在使用java8lambda表达式的时候,有时候会遇到这样的编译报错
Variable used in lambda expression should be final or effectively final_第1张图片
这句话的意思是,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();
    }

}

你可能感兴趣的:(java,jvm,java,开发语言)