Java中的迭代器和foreach原理&foreach循环报NullPointerException空指针异常

文章讲述了在Java编程中遇到的NullPointerException问题,特别是在使用foreach循环遍历集合时。当集合为null时,foreach会抛出异常。文章强调了应先检查集合是否为null,推荐使用ApacheCollectionUtils工具类的isEmpty方法。同时,解释了Java中的迭代器和foreach循环的工作原理,包括Iterator接口的方法以及foreach循环的底层实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

foreach循环报NullPointerException空指针异常

一、问题

最近写了个接口,一直报Exception in thread “main” java.lang.NullPointerException,定位到报错的语句,发现是一个foreach语句报错:

img

img

于是打个断点查看,发现 userInfoBaseBOList = null,

img

难怪一直报空指针。

二、说明

  • 对于空的集合,foreach循环会自动跳过对该空集合的遍历,但是不会报NPE;
  • 对于null值的集合,foreach循环则会直接报NPE。

img

img

img

不会报NPE

三、关于List集合为空和null的理解

  • list为空,实际上已经实例化对象开辟内存空间了,但是还没有保存数据;
  • list等于null,可理解为没有对list集合分配内存空间,实际上压根就不存在。

四、启示

用foreach遍历集合时,应该先使用Apache的工具类CollectionUtils判断集合是否为null,再进行遍历。

CollectionUtils的isEmpty()方法及isNotEmpty()方法源码如下:

img

if(list != null && !list.isEmpty()){ }



Java中的迭代器和foreach原理

迭代器是一种设计模式,它的定义为:提供一种方法访问一个容器对象中的各个元素,而又不需暴露该容器对象的内部细节。迭代器模式,就是为容器而生。

在Java中,Iterator称为迭代器,主要用于遍历 Collection 集合中的元素。Iterator 仅用于遍历集合,Iterator 本身并不提供承装对象的能力。如果需要创建Iterator 对象,则必须有一个被迭代的集合。Collection接口继承了java.lang.Iterable接口,该接口有一个iterator()方法,那么所有实现了Collection接口的集合类都有一个iterator()方法,用以返回一个实现了Iterator接口的对象。并且集合对象每次调用iterator()方法都得到一个全新的迭代器对象,默认指针(游标)都在集合的第一个元素之前。

1、Iterator接口中的常用方法

boolean hasNext():判断集合中是否还有下一个元素
E next():返回下一个元素
void remove():删除元素

注意:

  • 1.在调用next()方法之前必须要调用hasNext()方法进行判断。如果下一条记录无效,而且又没有调用hasNext()方法进行判断,那么直接调用next()方法就会抛出NoSuchElementException异常。
  • 2.Iterator可以删除集合的元素,但是是通过迭代器对象的remove方法,不是集合对象的remove方法。
  • 3.如果还未调用next()方法或在上一次调用next()方法之后已经调用了remove()方法,再调用remove()方法都会报IllegalStateException异常。

2.Iterator的使用方法(遍历集合)

Collection coll=new ArrayList();
coll.add(123);
coll.add(456);
coll.add(789);
coll.add(new Person("Tom",12));

Iterator iterator = coll.iterator();
while(iterator.hasNext()){
 System.out.println(iterator.next());
}

3、Iterator的执行原理

迭代器的执行原理如下图所示:

在这里插入图片描述

集合对象每次调用iterator()方法得到的迭代器对象,默认指针(游标)都在集合的第一个元素之前。我们在使用迭代器时,首先会使用hasNext()方法判断集合中是否还有下一个元素,如果有就会调用next()方法,调用next()方法分为两个步骤:一是把当前指针下移一位,二是将下移以后位置上的元素返回,如上图所示。

4、增强for循环:foreach

在Java 5.0提供了一种新的迭代访问 Collection和数组的方法,就是foreach循环。使用foreach循环执行遍历操作不需获取Collection或数组的长度,也不需要使用索引访问元素。
使用方法如下所示:

在这里插入图片描述

public static void main(String[] args) {
 int[] ints={1,2,3,4};
 for (int i:ints) {
  System.out.println(i);
 }

 Collection coll = new ArrayList();
 coll.add(123);
 coll.add(456);
 coll.add(789);
 coll.add(new Person(1001, "Tom"));
 for (Object object:coll) {
  System.out.println(object);
 }
}

那么foreach循环是怎样实现的呢?
下面我们来看一下上述代码编译后的源码文件,如下所示:

在这里插入图片描述

编译后的源码如下:

public static void main(String[] args) {
  int[] ints = new int[]{1, 2, 3, 4};
  int[] var2 = ints;
  int var3 = ints.length;

  for(int var4 = 0; var4 < var3; ++var4) {
   int i = var2[var4];
   System.out.println(i);
  }

  Collection coll = new ArrayList();
  coll.add(123);
  coll.add(456);
  coll.add(789);
  coll.add(new Person(1001, "Tom"));
  Iterator var7 = coll.iterator();

  while(var7.hasNext()) {
   Object object = var7.next();
   System.out.println(object);
  }

 }

由编译后的源码可知,使用foreach循环遍历数组时,底层仍然是通过for循环实现的;而遍历集合时,底层是通过迭代器实现的。

<think>好的,用户问的是Javaforeach遍历List的问题。首先,我得确认用户对Java的基础了解程度。可能他们刚接触循环结构,或者想了解更高效的遍历方法。首先,我需要解释foreach循环的基本语法,特别是在List上的应用。 然后,想到foreach其实是Java增强for循环,底层使用的是迭代器。这点很重要,因为如果在遍历过程中修改List,比如删除元素,可能会抛出ConcurrentModificationException。需要提醒用户注意这一点。 接下来,可能需要比较foreach传统for循环的区别,比如索引访问顺序遍历的不同使用场景。用户可能想知道什么时候该用哪种方式,这时候举例子会比较好。 还要考虑Java 8引入的forEach()方法,属于Iterable接口的一部分,可以结合Lambda表达式使用。这也是一个常见的用法,特别是对于更简洁的代码函数式编程风格。 然后,检查是否有常见的错误或误区,比如在遍历时修改集合,或者性能方面的考虑。比如,对于ArrayListLinkedList,不同的遍历方式可能有性能差异,但在大多数情况下,foreach已经足够高效。 最后,整理一个结构清晰的回答,分点说明基本语法、底层原理、两种实现方式(传统foreachJava 8的forEach()),以及注意事项。这样用户能逐步理解,并且知道在实际开发中如何应用避免问题。</think>在Java中,使用`foreach`循环遍历List集合是一种简洁且常用的方式。以下是逐步说明: --- ### 一、基本语法 `foreach`循环增强for循环)的语法格式为: ```java for (元素类型 变量名 : 集合或数组) { // 操作代码 } ``` 例如遍历`List<String>`: ```java List<String> list = Arrays.asList("A", "B", "C"); for (String item : list) { System.out.println(item); } ``` --- ### 二、底层原理 1. **迭代器实现** `foreach`循环底层通过`Iterator`迭代器实现,等价于: ```java Iterator<String> it = list.iterator(); while (it.hasNext()) { String item = it.next(); System.out.println(item); } ``` 2. **只读性** 在遍历过程中**不能直接修改List结构**(如增删元素),否则会抛出`ConcurrentModificationException`异常。 --- ### 三、两种实现方式 #### 方式1:传统`foreach`循环 ```java List<Integer> numbers = List.of(1, 2, 3); for (Integer num : numbers) { System.out.println(num * 2); } ``` #### 方式2:Java 8+ 的`forEach()`方法 通过Lambda表达式简化: ```java List<Integer> numbers = List.of(1, 2, 3); numbers.forEach(num -> System.out.println(num * 2)); ``` --- ### 四、注意事项 1. **遍历时修改集合** 如果需要增删元素,应改用`Iterator`的`remove()`方法或`for`循环配合索引。 2. **性能差异** - 对`ArrayList`:`foreach`与传统`for`循环性能接近 - 对`LinkedList`:`foreach`比索引遍历更高效 3. **空集合处理** 如果List为`null`会抛出`NullPointerException`,建议先判空: ```java if (list != null) { for (String item : list) { ... } } ``` --- ### 五、适用场景 - 仅需顺序访问元素且无需修改集合时 - 需要简化代码并提高可读性时 通过理解这些细节,可以更安全高效地使用`foreach`遍历List集合
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Archie_java

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值