Java集合类ArrayList循环中删除特定元素

在项目开发中,我们可能往往需要动态的删除ArrayList中的一些元素。

一种错误的方式:

for(int i = 0 , len= list.size();i<len;++i){
  if(list.get(i)==XXX){
       list.remove(i);
  }
}


上面这种方式会抛出如下异常:

Exception in thread "main" java.lang.IndexOutOfBoundsException: Index: 3, Size: 3
    at java.util.ArrayList.RangeCheck(Unknown Source)
    at java.util.ArrayList.get(Unknown Source)
    at ListDemo.main(ListDemo.java:20)
因为你删除了元素,但是未改变迭代的下标,这样当迭代到最后一个的时候就会抛异常咯。


可以对上面的程序进行如下改进:


上面的代码就正确了。

下面我们再介绍一种方案:

for(int i = 0 , len= list.size();i<len;++i){
  if(list.get(i)==XXX){
       list.remove(i);
       --len;//减少一个
       --i;//多谢deny_guoshou指正,如果不加会出现评论1楼所说的情况。
 }
}



List接口内部实现了Iterator接口,提供开发者一个iterator()得到当前list对象的一个iterator对象。
    Iterator<String> sListIterator = list.iterator();
    while(sListIterator.hasNext()){
        String e = sListIterator.next();
        if(e.equals("3")){
        sListIterator.remove();
        }
    }


上面这种也是正确的,并推荐使用第二种方案。

两种方案实现原理都差多的,第二种只是jdk封装了下。

查看ArrayList源码会发现很多方法内部都是基于iterator接口实现的,所以推荐使用第二种方案。


练习:删除ArrayList<String>中包含数字的字符串

/*
	 * 练习:删除ArrayList<String>中包含数字字符串的元素
	 */
	public void deleteString1(ArrayList<String> al){
		//根据正则表达式生成正则对象
		Pattern pattern = Pattern.compile("\\d");
		for(int i=0;i<al.size();i++){
			//将正则对象和字符串相关联生成匹配器
			Matcher matcher = pattern.matcher(al.get(i));
			//在字符串中寻找目标字符串
			if(matcher.find()){
				//将含数字字符串的元素删除后需要将指针前移一个单位
				al.remove(i--);
			}
		}
	}
	/**
	 * 测试
	 */
	@Test
	public void deleteString(){
		ArrayList<String> al = new ArrayList<String>();
		al.add("123");
		al.add("abcl1");
		al.add("abc");
		al.add("abc11");
		deleteString1(al);
		System.out.println(al);
	}


你可能感兴趣的:(Java集合类ArrayList循环中删除特定元素)