Java易错知识点(1) - 关于ArrayList移除元素后剩下的元素会立即重排
帮一个网友解答问题时,发现这样一个易错知识点,现总结如下:
1、易错点:
ArrayList移除元素后,剩下的元素会立即重排,他的 size() 也会立即减小,在循环过程中容易出错。(拓展:延伸到所有的集合是否可行)
2、易错点分析:
在for循环遍历ArrayList时,在循环中移除元素后,集合的size()会立即减1
剩下的元素也会重新排列,被移除元素后面元素的下标会发生变化,即后面的元素小标会减1
此时在for循环中的第二个参数 i < list.size() 就不是原来集合的大小了,而是比上一次循环小1
而循环变量 i 的值还是正常递增
如果继续遍历集合,就容易漏掉某个元素
3、实例分析:
此实例的目的是:去除集合中带有#字符的url
(1)错误版本:
<span style="font-family:KaiTi_GB2312;font-size:18px;">import java.util.ArrayList;
public class GouLv {
public static void main(String[] args) throws Exception {
ArrayList<String> urls = new ArrayList<String>();
urls.add("http://www.baidu.com");
urls.add("http:/#/www.baidu.com");
urls.add("http://www#.zxitb.com");
urls.add("http://www.zxitb.com");
int index;
for (int i = 0; i < urls.size(); i++) {
index = -1;
String url = urls.get(i);
index = url.indexOf('#');
if(index != -1){
urls.remove(url);
}else {
System.out.println(url);
}
}
for (int i = 0; i < urls.size(); i++) {
System.out.println("去除#后的url : " + urls.get(i));
}
}
}
-----------------------------------------------
输出结果为:
http://www.baidu.com
http://www.zxitb.com
去除#后的url : http://www.baidu.com
去除#后的url : http://www#.zxitb.com
去除#后的url : http://www.zxitb.com
------------------------------------------------
拓展:
如果map中的元素这样放
urls.add("http://www.baidu.com");
urls.add("http:/#/www.baidu.com");
urls.add("http://www.zxitb.com");
urls.add("http://www#.zxitb.com");
那结果为
http://www.baidu.com
http://www.zxitb.com
去除#后的url : http://www.baidu.com
去除#后的url : http://www.zxitb.com
感觉上时达到了目的,其实不然</span>
(2)错误分析版本:
<span style="font-family:KaiTi_GB2312;font-size:18px;">import java.util.ArrayList;
public class GouLv {
public static void main(String[] args) throws Exception {
ArrayList<String> urls = new ArrayList<String>();
urls.add("http://www.baidu.com");
urls.add("http:/#/www.baidu.com");
urls.add("http://www#.zxitb.com");
urls.add("http://www.zxitb.com");
int index;
for (int i = 0; i < urls.size(); i++) {
index = -1;
System.out.println("循环次数===========" + (i+1));
String url = urls.get(i);
index = url.indexOf('#');
if(index != -1){
System.out.println("有#的url: " + url);
urls.remove(url);
System.out.println("移除元素后urls的size: " + urls.size());
for(int j = 0; j < urls.size(); j++){ //移除元素后打印urls中的元素
System.out.println("移除元素后urls中下标为【 " + j + " 】的元素" + urls.get(j));
}
}else {
System.out.println("没有#的url: " + url);
}
System.out.println("urls的大小: " + urls.size() + "=====下一次循环i的值将会为: " + (i+1));
}
for (int i = 0; i < urls.size(); i++) { //打印最终结果
System.out.println("去除#后的url : " + urls.get(i));
}
}
}
------------------------------------------------------------
结果:
循环次数===========1
没有#的url: http://www.baidu.com
urls的大小: 4=====下一次循环i的值将会为: 1
循环次数===========2
有#的url: http:/#/www.baidu.com
移除元素后urls的size: 3
移除元素后urls中下标为【 0 】的元素http://www.baidu.com
移除元素后urls中下标为【 1 】的元素http://www#.zxitb.com
移除元素后urls中下标为【 2 】的元素http://www.zxitb.com
urls的大小: 3=====下一次循环i的值将会为: 2
循环次数===========3
没有#的url: http://www.zxitb.com
urls的大小: 3=====下一次循环i的值将会为: 3
去除#后的url : http://www.baidu.com
去除#后的url : http://www#.zxitb.com
去除#后的url : http://www.zxitb.com</span>
分析:
此时第一次循环的是http://www.baidu.com,第二次循环的是http:/#/www.baidu.com
由于第二次循环移除了元素,所以size()减1就是3,剩下的元素也重新排列。而此时 i 为2,就是循环的新集合中的http://www.zxitb.com(第三次循环)
此时符合条件,循环结束
跳过了http://www#.zxitb.com的循环
更改思路:
移除元素后,修改循环中的第二个参数或者第三个参数。此时我在移除元素后把循环变量 i 的值减1
(3)正确版本:
<span style="font-family:KaiTi_GB2312;font-size:18px;">import java.util.ArrayList;
public class GouLv {
public static void main(String[] args) throws Exception {
ArrayList<String> urls = new ArrayList<String>();
urls.add("http://www.baidu.com");
urls.add("http:/#/www.baidu.com");
urls.add("http://www#.zxitb.com");
urls.add("http://www.zxitb.com");
int index;
for (int i = 0; i < urls.size(); i++) {
index = -1;
String url = urls.get(i);
index = url.indexOf('#');
if(index != -1){
System.out.println("有#的url: " + url);
urls.remove(url);
i-- ;
}else {
System.out.println("没有#的url: " + url);
}
}
for (int i = 0; i < urls.size(); i++) { //打印最终结果
System.out.println("去除#后的url : " + urls.get(i));
}
}
}
---------------------------------------------------------
结果:
没有#的url: http://www.baidu.com
有#的url: http:/#/www.baidu.com
有#的url: http://www#.zxitb.com
没有#的url: http://www.zxitb.com
去除#后的url : http://www.baidu.com
去除#后的url : http://www.zxitb.com</span>
OK ! 问题解决!
Java易错知识点(1) - 关于ArrayList移除元素后剩下的元素会立即重排的更多相关文章
- Java易错知识点(2) - 在读取Cookie时除了Key,Value是得不到其他信息的
全文总结: 在读取Cookie,然后操作时,除了getName(),getValue()外,不要妄图得到其他信息,如下方法不会得到值的: cookie.getMaxAge(); cookie.getD ...
- JavaScript易错知识点整理
前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...
- JavaScript 易错知识点整理
本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一些ES ...
- JavaScript易错知识点整理[转]
前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...
- JS易错知识点
JAVASCRIPT易错知识点整理 前言 本文是学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由 ...
- 【笔试题】Java 易错题精选
笔试题 Java 易错题精选 1.写出下列程序的运行结果( )String 不变性Java 值传递 public class Test { public static void main(String ...
- JavaScript易错知识点
JavaScript易错知识点整理1.变量作用域上方的函数作用域中声明并赋值了a,且在console之上,所以遵循就近原则输出a等于2. 上方的函数作用域中虽然声明并赋值了a,但位于console之下 ...
- java易错基础知识点
一. Switch 1.其能接受的数据类型有四个,char , byte, short, int2.Default 可放在switch中的任何一个地方,但只有给定的条件匹配不到时,才会执行3.Case ...
- [置顶] 单片机C语言易错知识点经验笔记
今天写这一篇文章并不是因为已经想好了一篇文章才写下来,而是我要将这一篇文章作为一个长期的笔记来写,我会一直更新.在进行单片机开发时,经常都会出现一些很不起眼的问题,这些问题其实都是很基础的c语言知识点 ...
随机推荐
- 九度OJ--Q1473
import java.util.ArrayList;import java.util.Scanner; /* * 题目描述: * 大家都知道,数据在计算机里中存储是以二进制的形式存储的. * 有一天 ...
- 对 a = [lambda : x for x in range(3)] 的理解
上面的语句创建了一个列表 a ,其中有三个元素,每个元素都是一个 lambda 匿名函数. >>> a = [lambda : x for x in range(3)] >&g ...
- BZOJ 4595 SHOI2015 激光发生器 射线,线段,偏转
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=4595 题意概述: 给出一条射线和N条线段,射线遇到线段会发生反射,令入射角alpha,出射 ...
- Java 实现一个带提醒的定时器
定时闹钟预览版EXE下载链接:https://files.cnblogs.com/files/rekent/ReadytoRelax_jar.zip 功能说明: 实现了一个休息提醒器,用户首先设定一个 ...
- 最短路径——Dijkstra算法以及二叉堆优化(含证明)
一般最短路径算法习惯性的分为两种:单源最短路径算法和全顶点之间最短路径.前者是计算出从一个点出发,到达所有其余可到达顶点的距离.后者是计算出图中所有点之间的路径距离. 单源最短路径 Dijkstra算 ...
- chromium源码阅读
linux下chromium的入口函数在文件:src/chrome/app/chrome_exe_main_aura.cc 中 int main(int argc, const char** argv ...
- 【Android入门】——模拟器的创建及常见问题汇总
[前言] 刚刚接触Android,第一门课我们就来创建一个模拟器.安卓模拟器,简称AVD(Android Virtual Device),是安卓运行的虚拟设备.有了他以后,我们就不需要在连着安卓手机进 ...
- php serialize讲解与json性能测试
[序列化的概念] 序列化是将对象状态转换为可保持或可传输的格式的过程.与序列化相对的是反序列化,它将流转换为对象.这两个过程结合起来,可以轻松地存储和传输数据. 将对象的状态信息转换为可以存储或传输的 ...
- shit element ui & form password validation
shit element ui & form password validation shit docs https://github.com/yiminghe/async-validator ...
- SQLAlchemy技术文档(中文版)(上)
在学习SQLAlchemy的过程中,好多时候需要查官方Tutorial,发现网上并没有完整的中文版,于是利用这两天空余时间粗略翻译了一下. 翻译效果很差....但也算是强迫自己通读一遍Tutorial ...