Java中list.foreach不能使用字符串拼接的问题

目录
  • list.foreach不能使用字符串拼接
    • 如图,不能使用String进行拼接
  • foreach循环中不能使用字符串拼接
    • 问题
    • 解决
    • 原理   
  • lambda表达式使用局部变量要用final

list.foreach不能使用字符串拼接

如图,不能使用String进行拼接

因为Lambda的本质实际上是匿名内部类,所以t必须是final类型(不过代码中的final可以省略),是不可以重新赋值的。

可以使用

final StringBuilder str = new StringBuilder("已选择:");

如图二

foreach循环中不能使用字符串拼接

问题

    @Test
    public void forEachTest(){
        String str = "中国你好!";
        List<String> list = Arrays.asList("a","b","c","d");
        list.forEach(item->{
         //编辑错误:Variable used in lambda expression should be final or effectively final
            str += item;
        });
        //可以使用增强for
//        for (String item : list) {
//            str += item;
//        }
        System.out.println("结果:" + str);
    }

该编译不通过的根本原因:Lambda表达式中的局部变量要使用final的问题。

因为String类型属于引用数据类型,String字符串有不可变的特性,String在进行字符串拼接时,每次都会指向不同的地址值,因此str变量不能被看作是一个final类型,也就不符合Lambda表达式的使用要求。

解决

使用StringBuffer或StringBuilder:

    @Test
    public void forEachTest(){
        //String str = "中国你好!";
        List<String> list = Arrays.asList("a","b","c","d");
        StringBuffer stringbuffer = new StringBuffer("中国你好!");
        list.forEach(item->{
            stringbuffer .append(item);
        });
        //可以使用增强for
//        for (String item : list) {
//            str += item;
//        }
        System.out.println("结果:" + sb);
    }

原理   

StringBuffer是一个引用数据类型,在进行append()时,只是修改了内容,并没有改变地址值,这个stringbuffer变量就在编译时看成了final类型的变量,因此可以使用。

Lambda表达式中的局部变量要使用final的问题

lambda表达式使用局部变量要用final

lambda表达式本身是一个匿名内部类的一种编写形式,可以操作外部的变量

使用实例变量或静态变量是没有限制的(可认为是通过 final 类型的局部变量 this 来引用前两者)

使用局部变量必须显式的声明为 final 或实际效果的的 final 类型,即该变量从未被改变过

    @Test
    public void finalTest(){
        String str = "中国你好!";
        //在Lambda中使用该变量,该变量不能被修改过,java8会默认加上final
        //str = "c";
        List<String> list = Arrays.asList("a","b","c","d");
        List<String> collect = list.stream().filter(item -> {
            return item.equals(str);
        }).collect(Collectors.toList());
        System.out.println("结果:" + collect);
  //不能改变str,否则Lambda表达式中编译失败
  //str = "山东你好";
    }

一个局部变量如果要在匿名类或是 Lambda 表达式中访问,那么这个局部变量必须是 final 的,即使没有修饰为 final 类型,编译器也会自动加上 final 修饰符。

在 Java 8 下,即使局部变量未声明为 final 类型,一旦在 Lambda 表达式(匿名类) 中使用,就被强型加上了 final 属性,所以后面就无法再次给 str 赋值了。

为什么 Lambda 表达式(匿名类) 不能访问非 final 的局部变量呢?

因为实例变量存在堆中,而局部变量是在栈上分配,Lambda 表达式(匿名类) 会在另一个线程中执行。如果在线程中要直接访问一个局部变量,可能线程执行时该局部变量已经被销毁了,而 final 类型的局部变量在 Lambda 表达式(匿名类) 中其实是局部变量的一个拷贝。

在java编译时,匿名内部类也会被当作普通的类处理,只不过编译器生成它构造方法的时候,除了将外部类的引用传递了过来,还将基本数据类型的变量复制了一份过来,并把引用数据类型的变量引用也传递了过来。因此,基本数据类型的变量当然不能修改了,不然就会跟外部的变量产生不一致,这样的话变量的传递也就变得毫无意义了。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Java8 ArrayList之forEach的使用

    目录 Java8 ArrayList之forEach使用 一.用法 二.效率 ArrayList在foreach中remove的问题分析 iterator itr.hasNext 和 itr.next 实现 倒数第二个元素的特殊 如何避坑 方法一,还是fori,位置前挪了减回去就行了, remove后i--: 方法二,不用ArrayList的remove方法,用Itr自己定义的remove方法,代码如下: Java8 ArrayList之forEach使用 之前使用Java8.顺便整理自己学到的

  • java 使用foreach遍历集合元素的实例

    java 使用foreach遍历集合元素的实例 1 代码示例 import java.util.*; public class ForeachTest { public static void main(String[] args) { // 创建集合.添加元素的代码与前一个程序相同 Collection books = new HashSet(); books.add(new String("book1")); books.add(new String("book2&quo

  • java程序中foreach用法示例

    语法 复制代码 代码如下: for (Object objectname : preArrayList(一个Object对象的列表)) {} 示例 复制代码 代码如下: package com.kuaff.jdk5;import java.util.*; import java.util.Collection; public class Foreach{private Collection c = null; private String[] belle = new String[4]; pub

  • java8 forEach结合Lambda表达式遍历 List操作

    我就废话不多说了,大家还是直接看代码吧~ @Test void testJava8ForeachMap() { Map<String, Integer> items = new HashMap<>(); items.put("A", 10); items.put("B", 20); items.put("C", 30); items.put("D", 40); items.put("E&quo

  • Java中list.foreach不能使用字符串拼接的问题

    目录 list.foreach不能使用字符串拼接 如图,不能使用String进行拼接 foreach循环中不能使用字符串拼接 问题 解决 原理    lambda表达式使用局部变量要用final list.foreach不能使用字符串拼接 如图,不能使用String进行拼接 因为Lambda的本质实际上是匿名内部类,所以t必须是final类型(不过代码中的final可以省略),是不可以重新赋值的. 可以使用 final StringBuilder str = new StringBuilder(

  • vue中如何实现变量和字符串拼接

    整理文档,搜刮出一个vue中如何实现变量和字符串拼接的代码,稍微整理精简一下做下分享. 在data中定义变量: data() { return { a: 'A' } } 如何通过按钮点击实现字符串和变量a的拼接呢? <button @click='showMsg'></button> //vue methods: { showMsg() { alert(`获取了${a}`); } } 注意alert()里不是单引号,而是两个 ` 号(esc下面的按键). 点击按钮出现 以上就是本文

  • java中建立0-10m的消息(字符串)实现方法

    直接用StringBuilder,它的append方法方便快速构建字符串. StringBuilder sb1=new StringBuilder(); for(int i=0;i<1024*1024*10;i++){ sb1.append('a'+""); } 取消息时 String str=sb1.tostring(); 取1M str.substring(0, 1024*1024)).getBytes(); 以上这篇java中建立0-10m的消息(字符串)实现方法就是小编分

  • java中简单的截取分割字符串实例

    目前整理出来三种形式: 可根据实际需要选择使用 String userNameUrl; int beginIndex = 0; int endIndex = 0; userNameUrl = "454512@hongri@4944115455d9591b274648a06303d910de"; /** * 方法一: */ beginIndex = userNameUrl.indexOf("@")+1; endIndex = userNameUrl.lastIndex

  • 详解java中String值为空字符串与null的判断方法

    Java空字符串与null的区别 1.类型 null表示的是一个对象的值,而不是一个字符串.例如声明一个对象的引用,String a=null. ""表示的是一个空字符串,也就是说它的长度为0.例如声明一个字符串String s="". 2.内存分配 String a=null:表示声明一个字符串对象的引用,但指向为null,也就是说还没有指向任何的内存空间. String s="":表示声明一个字符串类型的引用,其值为""空

  • 面试题:java中为什么foreach中不允许对元素进行add和remove

    目录 1.foreach遍历ArrayList过程中使用 add 和 remove 2.追根溯源 2.1.modCount是什么? 2.2.expectedModCount 是什么? 2.3.熟悉的checkForComodification方法 2.4.流程回顾 3.避免fail-fast 机制 3.1.使用listIterator或iterator 3.2.使用CopyOnWriteArrayList 3.2.1.CopyOnWriteArrayList的add方法 3.2.2.CopyOn

  • java中list.forEach()和list.stream().forEach()区别

    目录 概述 区别 首先,它们的功能都是遍历数组每个元素并执行入参的accept()方法,但是它们的实现方式却不一样,在一些特定的情况下,执行会出现不同的结果. 在大多数情况下,两者都会产生相同的结果,但是,我们会看到一些微妙的差异. 概述 首先,创建一个迭代列表: List<String> list = Arrays.asList("A","B","C","D"); 最直接的方法是使用增强的for循环: for(S

  • Java中String的split切割字符串方法实例及扩展

    目录 一.public String[] split(String regex) 二.public String[] split(String regex, int limit) 三.扩展 总结 一.public String[] split(String regex) public String[] split(String regex): 根据传入的字符串参数,作为规则,切割当前字符串 String a="198,168,10,1"; String [] arr=a.split(&

  • 详解JAVA中的for-each循环与迭代

    在学习java中的collection时注意到,collection层次的根接口Collection实现了Iterable<T>接口(位于java.lang包中),实现这个接口允许对象成为 "foreach" 语句的目标,而此接口中的唯一方法,实现的就是返回一个在一组 T 类型的元素上进行迭代的迭代器. 一.迭代器Iterator 接口:Iterator<T> public interface Iterator<E>{ boolean hasNext

  • 详解.NET中string与StringBuilder在字符串拼接功能上的比较

    string与StringBuilder的在字符串拼接时执行效率上有差异,因为StringBuilder类中用了一个技巧:它申请了两倍的内存空间存放字符串,在调用Append方法拼接字符串时,会先检查剩余的空间是否能放下要拼接的字符串,若能放下,则将要拼接的字符串Copy到剩余的空间中,若不能放下,则再申请拼接后的字符串两倍的长度空间,将当前字符串Copy到新的空间中(除了两倍的空间外,这点跟string的拼接没有太多的差异).因此StringBuilder能提高字符串拼接的效率在于它减少了申请

随机推荐