随着java8的普及,越来越多的人开始使用lambda表达式。然后初学者们总会遇到一个奇怪的问题:
Variable used in lambda expression should be final or effectively final
这个错的产生的原因是因为在lambda表达式中访问外部变量时,都不允许有修改变量的倾向,即
1、变量必须是final类型的
2、如果没有定义成final,那么变量在初始化以后,不允许再有任何赋值的情况出现。(其实就是隐式final限制)
这种情况的具体原因本文不讨论,百度一大堆解释,本次主要讨论如果有非final变量,如何在lambda中使用。
方法一:
这是我最初遇到该类问题时的解决办法,就是不用lambda,改为普通写法。如将forecah循环改为Iterator模式,这种方法一劳永逸。
方法二:
利用list特性,地址不变,value可变。
List<User> a=new ArrayList<>();
boolean flag=true;
Set<Boolean> flagList =new HashSet<>();
a.forEach(item->{
if(item.getAge()>18) {
//Local variable flag defined in an enclosing scope must be final or effectively final
//flag=false;
flagList.add(false);
}
if(item.getName().equals("zhangsan")) {
flagList.add(false);
}
});
if(flagList.contains(false)) {
return false;
}
如上,可能例子不太好。之前公司的代码是,要对list中所有的对象进行规则校验,20多种情况。在写完以后才发现需要返回是否全部通过,这时改成迭代器模式就太麻烦了,于是就采用了这种方式返回。
方法三:
有时我们会遇到另一种情况,定义一个value=0,当type=1的时候,value=100。并且这种情况不需要在foreach中反复处理,可以采用封装方法的方式实现。
public static void main(String[] args) {
List<User> a=new ArrayList<>();
int value=0;
if(true) {
value=100;
}
int lambdaValue=getValue(true);
a.forEach(item->{
//Local variable flag defined in an enclosing scope must be final or effectively final
// item.setAge(value);
item.setAge(lambdaValue);
});
}
static int getValue(boolean flag) {
return flag?100:0;
}
方法四:
这个方法是方法三的变种,就是在之前处理完value后,将value再赋值给lambdaValue。
List<User> a=new ArrayList<>();
int value=0;
if(true) {
value=100;
}
int lambdaValue=value;
a.forEach(item->{
//Local variable flag defined in an enclosing scope must be final or effectively final
// item.setAge(value);
item.setAge(lambdaValue);
});
方法三和方法四可以根据实际情况来选择。
以上是个人工作中总结的一些方法,以后遇到新的方法会来补充,如果有人有更好的解决办法,也请指出,谢谢。