AngularJS:错误:[$ rootScope:infdig] - AngularJS视图中的调用函数

时间:2018-01-30 14:43:28

标签: javascript angularjs angularjs-scope angular-http angularjs-infdig

我写了一个包含这个功能的控制器:

$scope.getItemName = function (id) {
    sgcdata.get('item', {
        _id: id
    }).then(function (dataItem) {
        return dataItem[0].brand_name + " " + dataItem[0].reference;
    });
}

在视图中:

<li ng-repeat="itemCart in connectedUser.cart track by $index" class="list-group-item">
     <span class="badge cursorpointer" ng-click="removeFromCart(itemCart)">X</span>
     <a href="#">{{ getItemName(itemCart) }}</a>
</li>

当我这样做时,我得到一个无限循环($rootScope:infdig)。

sgcdata.get发出XHR请求。

但是如果我返回一个简单的字符串,它就可以了:

$scope.getItemName = function (id) {
    return "toto";
}

我真的不明白发生了什么......

感谢您的帮助。

修改

我制作了一个过滤器而不是之前的功能:

JS:

sgc.filter('getItemName', [
    'sgcdata',
    function (sgcdata) {
        return function (id) {
            sgcdata.get('item', {
                _id: id
            }).then(function (dataItem) {
                return dataItem[0].brand_name + " " + dataItem[0].reference;
            });
        }
    }
]);

HTML:

<li ng-repeat="itemCart in connectedUser.cart track by $index" class="list-group-item">
    <span class="badge cursorpointer" ng-click="removeFromCart(itemCart)">X</span>
    <a href="#">{{ itemCart | getItemName }}</a>
</li>

我没有任何无限循环,但是我无法在ng-repeat中获得多个值,而过滤器返回在项目请求中。

修改

sgcdata.get:

service.get = function (collection, filters) {
    var deferred = $q.defer();
    var formatRequest = function (obj) {
        var str = [];
        for (var p in obj) {
            str.push(encodeURIComponent(p) + '=' + encodeURIComponent(obj[p]));
        }
        if (str.length) {
            return '&' + str.join('&');
        } else {
            return '';
        }
    };
    $http({
        method: 'GET',
        url: '/v1/get/?element=' + collection.toLowerCase() + formatRequest(filters)
    }).then(function (response) {
        deferred.resolve(response.data);
    }, function (response) {
        console.error("[ FAILED ] >> " + response);
    });
    return deferred.promise;
}

2 个答案:

答案 0 :(得分:1)

您应该从函数返回相同的引用或值。否则事情就会失去控制。

答案 1 :(得分:1)

ng-repeat HTML模板中调用函数将始终导致Infinite Digest错误,因为摘要周期本身的性质。

实质上,ng-repeat尝试渲染子节点。在子节点中,您正在调用一个函数,这会导致摘要发生。这个新摘要导致ng-repeat重新开始,这会导致子节点再次运行,这会导致另一个摘要,无限期。

有一些普遍接受的方法可以解决这个问题,每个方法各有利弊。

  1. 从服务器获取原始数组时,请在服务器上执行逻辑以在原始返回值中包含所有相关字段。这是非常高效的,因为您允许服务器完成所有繁重的工作,并且只有一次调用服务器。但是,您可能会发送仅对应用程序的一小部分有用的额外数据,这可能会减慢初始启动时间。

  2. 在获取原始数组时,为返回的每个元素运行该函数,以将额外数据作为新元素添加到每个元素上。这是相当有效的,因为它意味着您的所有数据都是前载的,并且您只需要为每个元素进行一次额外的调用。这可能会受到与发送不常用数据相同的负面影响,如果您有许多要迭代的元素,这也会放大初始启动减速。

  3. ng-repeat添加过滤器,而不是元素,这些元素遍历数组并为ng-repeat提取独占的额外数据。这样做的好处是只在ng-repeat存在时才对服务器进行额外调用,这不会将额外数据添加到原始数组中。但是,这些对服务器的额外调用将​​在ng-repeat更改(添加新元素,删除项目,应用其他过滤器等)时执行,这可能会导致严重的性能损失,具体取决于数据量和变化的频率。