首页 > 解决方案 > 如何防止在 NgRx 中第二次加载数据?

问题描述

我有一个 DOM 块:

<div (click)="toggle(block)">
   <div *ngIf="block.opened">
     <div *ngFor="let item of block.items"></div>
   </div>
</div>

通过单击我调度操作以从服务器加载数据并在块中显示此数据:

  toggle(block: RegistryGroup) {
    this.store.dispatch(RegistryActions.ToggleRegistryBlockAction(block));
    this.store.dispatch(RegistryActions.LoadRegistryLayersAction(this.block));
  }

第一个事件ToggleRegistryBlockAction将属性更改block为真/假。第二个LoadRegistryLayersAction通过向服务器发出请求block.id并将数据返回block到与属性相同的状态block.items = Server Response;

在我听动作的有效方面LoadRegistryLayersAction

loadRegistriesLayers$: Observable<Action> = createEffect(() =>
    this.actions$.pipe(
      ofType(RegistryActions.LoadRegistryLayersAction),
      filter(
        (registryGroup) =>
          registryGroup.items && !registryGroup.items.length
      ),
      switchMap((registryGroup: RegistryGroup) =>
        from(this.registry.getRegistryLayers(registryGroup.Id)).pipe(
          map((layers: { [key: string]: RegistryLayerItemGeneric[] }) => {
            return RegistryActions.SuccessLoadRegistryLayersAction({
              payload: {
                layers: this.registry.createLayersMapWithObjects(layers),
                registryGroupId: registryGroup.Id,
              },
            });
          }),
          catchError((error: Error) => {
            return of(RegistryActions.ErrorRegistryLayersAction(error));
          })
        )
      )
    )
  );

filter采取行动并检查数据是否已经存在。为此,我检查块是否有registryGroup.items && !registryGroup.items.length.

问题是,有时执行此效果服务器后可能会返回 empty []。所以下一次,当用户请求它时,它会重复发送请求。

如果加载了数据,如何修复它并且不要触摸服务器?

标签: angularreduxngrxngrx-storengrx-effects

解决方案


有多种解决方案:

  • 保留一个状态属性,它可以是loading, loaded, empty,...
  • 保留在状态中获取的 id 列表,并在发送请求之前检查此列表
  • 不要检查registryGroup.items.length,并将数组初始化为null

推荐阅读