我是一个使用Redux(带有@ ngrx/store)的Angular 2应用程序,我以这种方式对商店进行了建模:

{

  modelA: {
    ids: [1, 2],
    entities: { 1: { name: "name modelA 1" },
                2: { name: "name modelA 2" }
              }
  },

  modelB: {
    ids: [5, 8],
    entities: { 5: { name: "name modelB 5" },
                8: { name: "name modelA 8" },
                9: { name: "name modelA 9" }
              }
  }

}

基本上,我有两种类型的对象:modelA和modelB。现在还可以。
但是我找不到哪种最好的方式来写之间的关系,代表诸如modelA之类的东西有许多modelB(一对多)。我可以做这样的事情吗?
modelAmodelB: {
  entities: {
    1: [5],
    2: [8, 9]
  }
}

这是商店的根,不是'modelA'的 child 。
这可能有效,但是如何使用@ ngrx/store方法从特定的modelA中“查询” modelB?因为如果我编写一个选择器函数来读取全局状态并从modelAmodelB返回部分状态,那么在编写函数时我无权访问其余状态。例子:
compose(getModelAEntities(number[]), getModelAModelBRelations(modelA_id: number), getModelAModelBState() );

我可以使用Observable.combineLast进行查询
Observable
.combineLatest(
  this.store.select('contentContents'),
  this.store.select('contents'),
  (relations: any, contents: any) => {
    return relations.entities[1].map( id => {
      return contents.entities[id]
    })
  }
).subscribe( data => {
  console.log(data);
})

但是我不知道这是否对:每当我更改modelA实体对象(例如,添加一个新对象)时,都会调用subscribe(),但是输出是相同的,因为既没有更改modelA实体也没有更改其modelB相关对象。

PS:我可以这样查询
export const getModelAModelBs = (id) => {
  return state =>
    state.select( (s: any) => [s.modelAModelB.entities[id], s.modelB.entities])
    .distinctUntilChanged( (prev, next) => {

      const new_m = (next[0] || []).map( id => {
        return next[1][id];
      });

      const old_m = (next[0] || []).map( id => {
        return prev[1][id];
      });

      return prev[0] === next[0] && (JSON.stringify(new_m) === JSON.stringify(old_m))
    })
    .map( ([ids = [], modelBs]) => ids.map( id => modelBs[id]) );
};

//use the selector
this.store.let(getModelAModelBs(1)).subscribe( data => {
  console.log(data)
})

但是我不知道这是否是最好的方法。

最佳答案

我在同一个问题上苦苦挣扎,并提出了包装器的解决方案。

请看一下ngrx-entity-relationship库:https://www.npmjs.com/package/ngrx-entity-relationship

您可以创建如下选择器:

export const selectUserWithCompany = entityUser(
  entityUserCompany(),
);

然后在商店中使用它

this.store.select(selectUserWithCompany, 'userId');

要得到

{
  id: 'userId',
  companyId: 'companyId',
  company: {
    id: 'companyId',
    // ...,
  },
  // ...,
}

关于javascript - @ Ngrx/商店: how to query models with relationships,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/44327028/

10-15 22:53