JAVA8 lambdas表达式 改变外面变量的值
如图,我在外部定义了一个etotalPrice,然后再两个for循环里想要改变这个值,结果给我报错了,该怎么解决?
(NumberUtil.add和mutiplyu是基础的+*保留2位小数的方法)
回答:
在Java的经典著作《Effective Java》、《Java Concurrency in Practice》里,大神们都提到:匿名函数里的变量引用,也叫做变量引用泄露,会导致线程安全问题,因此在Java8之前,如果在匿名类内部引用函数局部变量,必须将其声明为final,即不可变对象。
Java8这里加了一个语法糖:在lambda表达式以及匿名类内部,如果引用某局部变量,则直接将其视为final。
建议你重构一下这段代码:使用lambda返回一个值,赋给外面的变量。
回答:
【更新】忘记回答原始的问题了。
如果要在Lambdas里更改外部变量,一般有两种方法:
定义一个长度为1的数组
final double[] totalPrice = new double[];// lambdas 块:
{
totalPrice[0] += // TODO
}
定义Holder类,或着用MutableDouble
public class Holder<T> { private T value;
Holder(T value) {
this.value = value;
}
public static <T> Holder<T> of(T value) {
return new Holder<>(value);
}
public T getValue() {
return value;
}
public void setValue(T value) {
this.value = value;
}
// ... TODO
}
如果我没有搞错的话,你问题代码用lambda更标准的写法是:
puchasePrices.filter(skuVo -> skuVo.getCode() == code).mapToDouble(skuVo -> count * skuVo.getPurchasePrice()).sum();
完整的写法
Map<String, Integer> codeMap = ...;codeMap.entrySet().stream().mapToDouble(e -> puchasePrices.filter(skuVo -> skuVo.getCode() == e.getKey()).mapToDouble(skuVo -> e.getValue() * skuVo.getPurchasePrice()).sum()).sum();
一种可能更好的写法(再一次,因为没有看到你全部的代码,我只能猜测)
puchasePrices.filter(skuVo -> codeMap.containsKey(skuVo.getCode)).mapToDouble(skuVo -> codeMap.get(skuVo.getCode()) * skuVo.getPurchasePrice()).sum();
或者:
puchasePrices.mapToDouble(skuVo -> codeMap.getOrDefault(skuVo.getCode(), 0) * skuVo.getPurchasePrice()).sum();
建议你在原问题里粘贴代码,图片别人没法拷贝。
顺便再提一个建议,根据你贴出来的代码,puchasePrices在我看来好像该是orderItems什么之类,变量名是很重要,如果是我弄错了,请原谅。
回答:
意思是totalPrice在lambada表达式中,应该是final类型,final类型初始化后就不可以改变了,所以再次给totalPrice赋值会错误。所以应该重新定义一个变量保存新的值,而不是再次将值复制给totalPrice,如果改变变量不可以,就不要使用lambada表达式。
回答:
@泊浮目 已经将原因说得很清楚了,给你提供一个变相的解决方式。使用数组或者集合(不推荐,仅供参考)
public static void main(String[] args) { int[] sum = new int[1];
List<Integer> nums = new ArrayList<Integer>(){{
add(1);
add(2);
}};
nums.stream().forEach(n -> sum[0]+=n);
}
回答:
你这段代码是用函数式的接口写命令式的程序,那为什么不用for循环做呢?
如果采用函数式思路的话,大致写法如下
Double totalPrice = codeMap.map( (code, count) -> {
return NumberUtil.multiply(count,getPrice(code),2);
}).reduce(0.0, (sum, eachPrice) -> {
return NumberUtil.add(sum, eachPrice, 2)
});
回答:
你可以将你想要改变的值放到一个类的实例中做为属性,在 lambda 中修改实例中的属性:
PriceObj price=new PriceObj();codeMap.forEach( (code, count) -> {
price.totalPrice=xxxxx;
})
类似这样
回答:
final 的,当然不可变啦。如果必须要变就不要用lambda,用lambda就别修改值
以上是 JAVA8 lambdas表达式 改变外面变量的值 的全部内容, 来源链接: utcz.com/p/180979.html