瘋狂

          STANDING ON THE SHOULDERS OF GIANTS
          posts - 481, comments - 486, trackbacks - 0, articles - 1
            BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理
             考慮以下代碼:此問題以前在javaeye上看見過,不過在我們的代碼評審的時候仍然發現相同問題,代碼看著很正常,但結果卻差之千里。
          目的:刪除list里面的所有元素。
          List list = new ArrayList();   
              list.add("1");   
              list.add("2");   
              list.add("3");   
              list.add("4");   
              list.add("5");   
              list.add("6");    
              for (int i = 0; i < list.size(); i++) {    
                  list.remove(i);   
              }
          for (int i = 0; i < list.size(); i++) {   

                      System.out.println(list.get(i));   

                  } 


          結果:
          2
          4
          6
          為什么沒刪除完,分析一下:
          當在刪除的過程中,i 在變大,但是list中的元素的位置則在變小,例如:

          當刪掉第一個元素的時候,第二個元素的index立馬變成o,而這時候i卻成了1 這就導致index為0的元素無法刪除。
          這種情況問題很難發現。
          但有的人喜歡這樣寫,到是能及時發現問題:
           int n = list.size();
              for (int i = 0; i < n; i++) {   
                  list.remove(i);   
              }
          這個時候就會拋出 java.lang.IndexOutOfBoundsException異常,原因很明顯,list的size是在每次刪除的時候都減1,而n卻一直沒變,最終導致越界。
          當然有的人提出改進的方法:
          for (int i = list.size() - 1; i >= 0; i--) {   

                      list.remove(i);   

                  }

          這樣確實能全部刪除,但是存在僥幸。因為和我們實際的想法有悖的,我們的想法是從大到小刪除,但是實際上發生的事情確實我們總是在刪除最大的。分析一下:
          當刪除list中index最大的元素后,原來倒數第二個元素變為最大,但是index變為原來的size-1,通過i也變成size-1。最終原來index為0 的元素變為index最大的元素。
           
            個人總結出的原則:1 盡量不要讓單個for循環中的兩個邊界發生變化,這樣我們可以對游標(i)的變化心里有數。
                                                  2  不要把游標(i)和不斷變化的資源位置掛鉤,這樣我們可以對資源的變化心里有數。
          關于list的刪除方法就不在這將了,不管用iterator,還是其他的
          總的來說,讓程序按照我們的設計步驟進行才是成功的。就像這幾天的嫦娥2號,要是發生這樣的事情,就慘了,本來要先點一級火箭,再二級,再三級,結果一級點了,就直接3級了,估計也沒法繞月了,這是多么大的損失啊。


          評論

          # re: 讓程序向衛星發射一樣受控:謹慎在for循環中的使用集合的remove(int index)   回復  更多評論   

          2010-10-08 09:33 by Chase
          這個確實沒注意到,LZ提醒了我,不錯不錯

          # re: 讓程序向衛星發射一樣受控:謹慎在for循環中的使用集合的remove(int index)   回復  更多評論   

          2010-10-08 23:32 by landas
          請使用Iterator.remove()

          # re: 讓程序向衛星發射一樣受控:謹慎在for循環中的使用集合的remove(int index)   回復  更多評論   

          2010-10-09 10:58 by @joe
          此文是方法論而非方法實現。

          # re: 讓程序向衛星發射一樣受控:謹慎在for循環中的使用集合的remove(int index)   回復  更多評論   

          2010-10-17 13:47 by liudecai
          提醒的好,前不久一個同事就這樣做了一把。他開始還以為遇上了靈異事件了。

          # re: 讓程序向衛星發射一樣受控:謹慎在for循環中的使用集合的remove(int index)   回復  更多評論   

          2013-11-18 16:28 by java菜
          for (int i = 0; i < list.size(); i++) {
          list.remove(i);
          i--;
          }
          主站蜘蛛池模板: 徐水县| 尼勒克县| 金溪县| 醴陵市| 苏尼特右旗| 法库县| 吉木乃县| 佛坪县| 灵寿县| 徐州市| 成武县| 巴东县| 镇安县| 安国市| 社会| 台东县| 客服| 房山区| 元朗区| 汉阴县| 平果县| 百色市| 胶州市| 黔西| 潞西市| 贵港市| 鄂托克前旗| 长海县| 会宁县| 梁平县| 太仆寺旗| 阳曲县| 河池市| 乌拉特后旗| 安西县| 辽阳县| 肥城市| 桑植县| 宜州市| 晴隆县| 新闻|