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语言知识点 ...
随机推荐
- swarm 服务器安装
apt install docker.io-----------------------配置加速器.私有仓库地址---------------------mkdir -p /etc/dockertee ...
- WebStorm强大的调试JavaScript功能(转载)
一.JavaScript的调试 目前火狐和Chrome都具备调试JavaScript的功能,而且还是相当的强大.如果纯粹是用浏览器来进行js调试的话,我比较喜欢用火狐.火狐可以安装各种插件,真的是非常 ...
- windows下自己常用的几个bat
1.samba映射盘符和解除 net use Z: "\\sambaserver ip\dir" "password" /user:"username ...
- Spark程序
Spark认识&环境搭建&运行第一个Spark程序 2017-07-09 17:17 by 牛仔裤的夏天, 181 阅读, 0 评论, 收藏, 编辑 摘要:Spark作为新一代大数据计 ...
- C# 托盘图标闪烁
在用户正在登录QQ或者使用Firemail邮件系统自动收取邮件的时候,托盘图标会闪动提示用户正在运行的任务.闪动图标可以使用定时切换托盘图标的方式实现,托盘图标可以从ImageList控件中获取.在I ...
- Div+Css中transparent制作奥运五环
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- ASP.NET页面之间传值Server.Transfer(4)
这个才可以说是面象对象开发所使用的方法,其使用Server.Transfer方法把流程从当前页面引导到另一个页面中,新的页面使用前一个页面的应答流,所以这个方法是完全面象对象的,简洁有效. Serve ...
- Codeforces Round #391 div1 757F (Dominator Tree)
首先先膜杜教orz 这里简单说一下支配树的概念 支配树是对一个有向图来讲的 规定一个起点s,如果s到v的路径上必须经过某些点u,那么离s最近的点u就是v的支配点 在树上的关系就是,v的父亲是u. 一般 ...
- 【题解】CQOI2015选数
这题做的时候接连想错了好多次……但是回到正轨上之后依然是一个套路题.(不过这题好像有比莫比乌斯反演更好的做法,莫比乌斯反演貌似是某种能过的暴力ヽ(´ー`)┌)不过能过也就行了吧哈哈. 首先我们把数字的 ...
- 【题解】NOI2014动物园
传送门:洛谷P2375 一直到写到这道题目才发现我一直都理解了假的KMP……fail数组:fail[i]为从1-i(包含i)在内的字符串,相同的最长前后缀长度. 那么我们可以先思考暴力:先求出所有的f ...