当前位置:网站首页>今日问题-2022/7/4 lambda体中修改String引用类型变量

今日问题-2022/7/4 lambda体中修改String引用类型变量

2022-07-06 17:46:00 机智小袁

说明:

1:lambda表达式的作用域中,对于局部引用变量是不做限制的。也就是可以在lambda体中对局部引用变量的值进行修改。原因是引用变量的值是保存在堆中的,是线程共享的所以Lambda可以修改他的值的。

遇到问题

String类型的变量是引用变量对吧,但是String类型的变量在lambda体中不允许修改。

public class Test {
    
    static String staticStr = "静态变量";
    String instanceStr = "实例变量";
    public static void main(String[] args) {
    
		/** * 遇到一个问题,String不算引用类型吗?为什么在lambda中修改String类型变量不成功。 */
        int t = 0;
        List<String> list = new ArrayList<>();
        list.add("我是第0个元素");
        String str = "我是局部引用变量";
        WorkerInterface workerInterface = (a,b)->{
    
            System.out.println("a:"+a+"b:"+b);
            list.get(0);
            list.add("eee");
            //此处会报错
            System.out.println(str);
            int c = t+1;
            System.out.println("c:"+c);
        };
        workerInterface.doSomeWork(3,4);
        list.add("33");
        str = str+"我改变了";
   }
}

/** * 自定义函数式接口 * @author wangdawei */
@FunctionalInterface
public interface WorkerInterface {
    
    /** * 一个抽象方法 */
    public void doSomeWork(int a,int b);
}

猜测

可能于String类型的优化有关,String类型是维护有一个字符串常量池的,池中的字符串会进行复用,池中的字符串是线程公有的。

解决

在lambda中允许修改局部引用类型变量的数据,但是不允许修改引用的指向。又因为String类型是不可变的类型,也就是说如果你要修改String类型变量的值,那就是将当前引用指向了其它地址。也就是改变了引用的指向,所以是错误的。
在这里插入图片描述

应对

如果想要使用那可以使用可变字符串

public class Test {
    
    static String staticStr = "静态变量";
    String instanceStr = "实例变量";
    public static void main(String[] args) {
    
		/** * 遇到一个问题,String不算引用类型吗?为什么在lambda中修改String类型变量不成功。 */
        int t = 0;
        List<String> list = new ArrayList<>();
        list.add("我是第0个元素");
        String str = "我是局部引用变量";
        WorkerInterface workerInterface = (a,b)->{
    
            System.out.println("a:"+a+"b:"+b);
            list.get(0);
            list.add("eee");
            stringBuffer.append("我变了");
            System.out.println(stringBuffer.toString());
            stringBuilder.append("我变了");
            System.out.println(stringBuilder.toString());
            int c = t+1;
            System.out.println("c:"+c);
        };
        workerInterface.doSomeWork(3,4);
        list.add("33");
   }
}
/** *结果: *我是可变字符串StringBuffer我变了 *我是可变字符串stringBuilder我变了 *c:1 **/

原网站

版权声明
本文为[机智小袁]所创,转载请带上原文链接,感谢
https://blog.csdn.net/wangdawei_/article/details/125609222