开发者社区> 技术mix呢> 正文

Java动态 遍历List 时删除List特征元素 异常问题 及解决方案总结

简介:
+关注继续查看

首先。这是一个极其简单的问题,大牛可忽略。新手可能会遇到,Java中遍历某个List 时删除该List元素 会抛出异常。


这一个简单的问题再高手严重不值一提,但新手可能会比較困惑,用哪种方式能够安全有效的实现遍历list删除某些特征元素?

方式:

方法1、普通for循环 遍历List,删List除自身 特征条目;

方法2、高级for循环 遍历List。删除List自身 特定条目;

方法3、引入參考List。for循环遍历删除原List 特定条目

方法4、利用iterator 遍历删除List特定条目


结论:

方法1:不会抛异常 但结果 未必正确 因为list.size()在变动 依据脚标删除的未必是 特征元素;

方法2:会抛异常  :java.util.ConcurrentModificationException

java.util.ConcurrentModificationException
at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:819)
at java.util.ArrayList$Itr.next(ArrayList.java:791)

方法3可实现目的,通过引入一个參考list for循环遍历參考list 在循环过程中 满足条件出 删除原list条目。能够达到所需目的

方法4可实现目的。通过list.iterator();方法拿到iterator对象 调用iterator.remove();方法可达到所需目的且不会抛异常。



这个问题 最好亲自执行代码。尝试,方能更好理解。


不废话。上代码,自己执行一下便可:

package com.kevin.test;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Random;

/**
 * 动态删除 List 特征元素 演示样例
 * 
 * @author blj
 * 
 */
public class TestMain
{

	static class Bean
	{

		private String name;

		public String getName()
		{
			return name;
		}

		public void setName(String name)
		{
			this.name = name;
		}

	}

	/**
	 * @param args
	 */
	public static void main(String[] args)
	{

		dynamicRemoveItemFromListDemo();
	}

	/**
	 * 动态删除List元素演示样例
	 */
	public static void dynamicRemoveItemFromListDemo()
	{

		List<Bean> list = getList();

		System.out.println("原始数据:");
		printList(list);

		// 使用普通for循环遍历List删除自身元素 不报错 但结果 未必正确
		normalRemoveMethod(list);

		// 使用高级for循环遍历List删除自身元素 报错
		// try
		// {
		// superRemoveMethod(list);
		// }
		// catch (Exception e)
		// {
		// System.err.println("高级for循环报错:");
		// e.printStackTrace();
		// }

		// 引入參考List 遍历删除方法
		// referenceRemoveMethod(list);

		// iterator遍历删除方法
		// iteratorRemoveMethod(list);

		System.out.println("结果数据:");
		printList(list);

	}

	/**
	 * 打印 List 方法
	 * 
	 * @param list
	 */
	private static void printList(List<Bean> list)
	{
		for (Bean item : list)
		{
			System.out.print(item.getName() + " ");
		}
		System.out.println();
	}

	/**
	 * 利用普通for循环遍历删除元素
	 * 
	 * @param list
	 */
	private static void normalRemoveMethod(List<Bean> list)
	{
		for (int i = 0; i < list.size(); i++)
		{
			if (list.get(i).getName().equals("A"))
			{
				list.remove(list.get(i));
			}
		}
	}

	/**
	 * 利用高级for循环遍历删除元素(会报异常)
	 * 
	 * @param list
	 */
	private static void superRemoveMethod(List<Bean> list)
	{
		// 方法一
		for (Bean item : list)
		{
			if (item.getName().equals("A"))
			{
				list.remove(item);
			}
		}
	}

	/**
	 * 有效的for循环遍历删除元素(正常 达到预期目的)
	 * 
	 * @param list
	 */
	private static void referenceRemoveMethod(List<Bean> list)
	{
		List<Bean> referenceList = new ArrayList<Bean>();
		referenceList.addAll(list);

		for (int i = 0; i < referenceList.size(); i++)
		{
			if (referenceList.get(i).getName().equals("A"))
			{
				list.remove(referenceList.get(i));
			}
		}
	}

	/**
	 * 使用Iterator的方式也能够顺利删除和遍历
	 */
	public static void iteratorRemoveMethod(List<Bean> list)
	{
		Iterator<Bean> it = list.iterator();
		while (it.hasNext())
		{
			Bean student = it.next();
			if (student.getName().equals("A"))
				it.remove();
		}
	}

	/**
	 * 获取List方法
	 * 
	 * @return
	 */
	private static List<Bean> getList()
	{
		List<Bean> list = new ArrayList<Bean>();

		for (int i = 0; i < 50; i++)
		{
			Bean item = new Bean();
			int j = i + new Random().nextInt();
			if (j % 3 == 1)
			{
				item.setName("A");
			}
			else if (j % 3 == 2)
			{
				item.setName("B");
			}
			else
			{
				item.setName("C");
			}

			list.add(item);
		}
		return list;
	}

}








本文转自mfrbuaa博客园博客,原文链接:http://www.cnblogs.com/mfrbuaa/p/5243239.html,如需转载请自行联系原作者

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
手撸一款简单高效的线程池(五)
在之前的内容中,我们给大家介绍了 C++实现线程池过程中的一些常用线优化方案,并分析了不同机制使用时的利弊。这一篇,是线程池系列的最后一章。我们会介绍一下 CGraph 中的 threadpool 如何使用,给出性能对比,并对接下来的工作做一些展望。让我们在线程池性能优化和功能提升的道路上,越走越远。
5 0
心中有“树”!图文并茂介绍数据结构中常见的树(一)
提到数据结构中的树(Tree) ,大家应该都不陌生,相关书籍中都有大段篇幅的介绍,刷 Leetcode 的时候会遇到很多相关问题。很多人往往会用 “手写红黑树” 来形容面试难度很高。
10 0
这个开源项目绝绝子,一键生成好玩的矢量风格头像!
最近逛 GitHub,发现了一个非常好玩的开源项目——头像生成器,给大家分享一下~
6 0
JSP - 起源、执行过程、运行原理、生命周期
JSP - 起源、执行过程、运行原理、生命周期
5 0
心中有“树”!图文并茂介绍数据结构中常见的树(三)
在前面两篇文章中,我们简要介绍了数据结构中的各种【树】在搜索、数据库等领域的使用场景,希望对大家有所帮助。
11 0
+关注
2968
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
OceanBase 入门到实战教程
立即下载
阿里云图数据库GDB,加速开启“图智”未来.ppt
立即下载
实时数仓Hologres技术实战一本通2.0版(下)
立即下载