0

我在我的项目中使用 NGRX,并且在不断地重新渲染组件时遇到了问题。商店每次都会产生全新的对象,如果商店数据相同,我不想重新渲染我的组件。我将表示组件的更改检测更改为ChangeDetectionStrategy.OnPushlodash memoization ( _.memoization),但它不起作用

店铺状况

export interface CardsState {
    cardsA: CardA[];
    cardsB: CardB[];
    cardsC: CardC[];
}

然后将 Store 状态映射为这样的结构

const entities = [{
 cards: {
  cardsB: [
   {}
  ]
 }
}];

并试图防止cardsB每次通过来自 lodash 的 memoize 更改存储时重新渲染项目

export const memoizeCardBFunc = _.memoize((cardB: CardB): CardB => {
  return _.cloneDeep(cardB);
}, (cardB: CardB) => cardB.entityType + cardB.entityId);

在我的智能组件中,我有下一个 html

<div *ngFor="let entity of entities | async">
    <div *ngFor="let cardB of entity.cards.cardsB">
        <card-b [cardStateModel]="cardB"></card-b>
    </div>
</div>

令我惊讶的是,它可以与一个 ngFor 一起正常工作,但如果我有两个 ngFor,它就不起作用了!

<div *ngFor="let cardB of entity.cards.cardsB">
    <card-b [cardStateModel]="cardB"></card-b>
</div>

我已经花了2天的时间,它正在杀死我,请帮助!

我不是以英语为母语的人,对错误深表歉意。

谢谢!

4

2 回答 2

1

在 For 循环中使用 trackBy。它将跟踪状态,并且当数据发生更改时,它将触发重新渲染。

    <div *ngFor="let cardB of entity.cards.cardsB;let i = index; trackBy: trackByFn">
        <card-b [cardStateModel]="cardB"></card-b>
    </div>

trackByFn(index, item) {
    return index; // or item.id
  }

阅读下面的文档,您会了解更多。 https://v2.angular.io/docs/ts/latest/api/common/index/NgFor-directive.html

于 2017-06-15T15:01:18.687 回答
1

如果你有:

<div *ngFor="let outer of outers">
    <div *ngFor="let inner of outer.inners">
       {{ inner }}
    </div>
</div>

然后,如果outer在某个时刻发生变化,内部 ngFor 将重新渲染,这就是为什么您会看到来自 ngOnInit 的警报。但是,如果您将 trackBy 添加到外部 ngFor,并且它会将其识别为相同的元素,那么您可以避免内部列表重新渲染,这是您当前的问题。

内部 trackBy 仅保留内部元素并防止重新创建这些元素。外部 trackBy 保留外部元素,这允许内部元素也有机会被保留。

于 2017-06-21T20:59:07.770 回答