用ng-repeat指令遍歷一個javascript數組,當數組中有重復元素的時候,angularjs會報錯,這是因為ng-Repeat不允許collection中存在兩個相同Id的對象。
對於數字或者字符串等基本數據類型來說,它的id就是它自身的值。因此數組中是不允許存在兩個相同的數字的。為了規避這個錯誤,需要定義自己的track by表達式。
// 業務上自己生成唯一的id
item in items track by item.id//或者直接拿循環的索引變量$index來用
item in items track by $index
Error: [ngRepeat:dupes]
這個出錯提示具體到題主的情況,意思是指數組中有2個以上的相同數字。ngRepeat不允許collection中存在兩個相同Id的對象
一個簡單動作(action)的列表
先來看看一個完整有效的ng-repeat
示例。
<ul ng-controller="ListCtrl">
<li ng-repeat="item in items">
{{item.name}}
<button ng-click="remove($index)">remove</button>
</li>
</ul>
對應的控制器(controller)如下:
app.controller('ListCtrl', ['$scope', function($scope) {
//items come from somewhere, from where doesn't matter for this example
$scope.items = getItems();
$scope.remove = function(index) {
var item = $scope.items[index];
removeItem(item);
};
}]);
看起來沒什么問題,對嗎? 這段代碼也沒有任何特別值得注意的。
添加一個過濾器(filter)
然后,讓我們來做一個小小的修改: 給列表添加一個過濾器。 這是很常見的做法,如果列表很長的話,例如允許用戶進行搜索。
為了方便起見, 假設我們通過 searchFilter
來查詢列表中的記錄。
<ul ng-controller="ListCtrl">
<li ng-repeat="item in items | searchFilter">
{{item.name}}
<button ng-click="remove($index)">remove</button>
</li>
</ul>
控制器的代碼保持不變。 看起來仍然沒有問題,是吧?
事實上,有一個bug藏在里面。 如果我不說, 你能找到嗎? 如果能找到,你就已經是Angular大牛了.
請盡量不要使用 $index
BUG其實是在控制器里面:
$scope.remove = function(index) {
var item = $scope.items[index];
removeItem(item);
};
這里使用了 index參數, 然后就遇到了BUG: 過濾后的索引(indexs)不匹配原始列表的索引。
幸運的是,有一個很簡單的方法來避免這種問題: 不要使用$index
,而改成實際的item對象。
<ul ng-controller="ListCtrl">
<li ng-repeat="item in items | searchFilter">
{{item.name}}
<button ng-click="remove(item)">remove</button>
</li>
</ul>
控制器如下所示:
$scope.remove = function(item) {
removeItem(item);
};
注意, 這里將 remove($index)
改成 remove(item)
, 並修改了 $scope.remove
函數來直接操作傳過來的對象。
這個小小的修改就完全避免了剛才的BUG。
為了更好地說明問題以及解決方案,請參考 interactive example 。
從中可以學到什么?
第一個教訓當然是在使用 $index
要小心一點,因為以某些方式使用時很可能會產生BUG。
第二個教訓是,請記住類似這樣的模式,則可以用更好的做事方式,可以完全避免某些類型的BUG。 我強烈建議大家現在不要使用 $index
, 從這種簡單的思維轉變中,就可以減少代碼中的很多BUG。
第三個教訓是測試並不是什么時候都有用。 即便有自動化測試,也覆蓋了足夠多的情形, 但對於依賴特定輸入的情況,也很容易錯過某些BUG。 錯誤本身並不是每次都會出現,即使你也用過濾來測試。
第四個教訓是不要破壞抽象 —— 這一點很容易被忽略。理論上 $index
是由 ng-repeat
創建的一個 “模板變量(template variable)”。 這只在 repeat 塊里面有意義(並正確起作用)。 當我們將它的值傳遞到外面時,它就失去了上下文從而不再有效。 如果確實想讓它在 repeat 之外依然有效,則必須在控制器中也進行過濾,這就需要一些不是很必要的重復代碼。 值得慶幸的是本文中介紹的模式可以用來避免這種情況。