• Java易错知识点(1)


    帮一个网友解答问题时,发现这样一个易错知识点,现总结如下:


    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 ! 问题解决!


  • 相关阅读:
    1、Elasticsearch教程-从入门到精通
    3、ik分词器
    2、ElasticSearch的安装
    svn操作
    2020-10
    编写第一个linux驱动
    字符设备驱动
    Linux下/dev和/sys/class的区别
    latex
    Qt
  • 原文地址:https://www.cnblogs.com/oldinaction/p/5167482.html
Copyright © 2020-2023  润新知