Redux 怎么处理嵌套的 selector?真的整糊涂了 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
sillydaddy
0.69D
0.11D
V2EX    React

Redux 怎么处理嵌套的 selector?真的整糊涂了

  •  
  •   sillydaddy 2022-08-18 09:48:58 +08:00 2350 次点击
    这是一个创建于 1152 天前的主题,其中的信息可能已经有所发展或是发生改变。

    我想渲染一棵树,RenderItem 是树的节点,可以不断嵌套子节点,构建好这棵树后,会把根节点 RenderItem 传给 react-sortable-tree 去渲染。

    type RenderItem = {id:string, name:string, children:RenderItem[]}; 

    这棵树的源数据,也是树状的,但是组织方式是扁平的:

    store: { "111": {name:"node1", children_ids:["222", "333"]}, "222": {name:"node2", children_ids:["444"]}, "333": {name:"node3", children_ids:["555"]}, "444": {name:"node4", children_ids:[]}, "555": {name:"node5", children_ids:[]}, } 

    当某些操作,修改了源数据里的一小部分,Redux 会利用 immer 生成新的 store 。

    这时候如果想要渲染某个节点,就要利用 Redux 的 useSelector ,根据新的 store 生成新的 RenderItem 树。

    问题的关键在于,如果我只修改了某个节点的很小一部分数据,就重新生成整个 RenderItem 树,会感觉效率很低。所以有没有办法使用 reselect 这种库,缓存节点的渲染数据呢?

    我试了试,发现很难,比如我想到的方案

    var selectNode = createSelector( (state, node_id)=>state[node_id], (state, node_id)=>node_id, (node, node_id)=>{ var childrenItems = node.children_ids.map((child_id)=>selectNode(state(从哪里来?), child_id)); var item:RenderItem = {id:node_id, name:node.name, children: childrenItems}; return item; } ) 

    从上面代码可以看到,selectNode 嵌套调用时,总是需要 state 这个参数的,而 state 又总是变化的,也就导致了整个树的数据都要重新生成。

    不知道我说清楚了没。这里不要纠结于 reselect 默认只能缓存 1 个数据,就假设它可以缓存很多个。

    10 条回复    2022-09-28 19:02:04 +08:00
    towry
        1
    towry  
       2022-08-18 10:07:27 +08:00
    你是通过 selector 选择后的数据来生成 RenderItem 树,对吗?
    sillydaddy
        2
    sillydaddy  
    OP
       2022-08-18 10:16:57 +08:00
    @towry
    selectNode 就是用来生成 RenderItem 树的,它接收 store 和 node_id 这 2 个参数,会把 node_id 对应的节点以及它的子节点递归生成出一棵 RenderItem 树。
    BingoXuan
        3
    BingoXuan  
       2022-08-18 10:47:11 +08:00
    我最近工作内容也和 op 的差不多。但我并没有用扁平的方法。而是标准的树,另外保存了节点名字。通过节点名字再从树搜索节点信息。渲染时候直接渲染树,通过 memo 来缓存值每一层的子树。
    kongkx
        4
    kongkx  
       2022-08-18 11:13:23 +08:00 via iPhone
    key 跟 object 分开处理。 再处理一下 key tree selector 的 compare 。
    GreatAuk
        5
    GreatAuk  
       2022-08-18 11:21:29 +08:00
    接收不了 redux, 样板代码太多
    sillydaddy
        6
    sillydaddy  
    OP
       2022-08-18 11:39:37 +08:00
    @kongkx
    可以细说一下吗,感谢。
    kongkx
        7
    kongkx  
       2022-08-18 12:45:54 +08:00 via iPhone
    @sillydaddy 做一个只包含层级关系的 key tree ( ID ) ,然后具体的节点渲染组件中通过 id ,select 对应节点的数据。 每次 state 修改的时候,还是会触发 keytree 的构建,那就通过自定义的 compare 函数来判断,前后数据是否 equal 。

    memoize 的库 一般都有 equality 的方法可以设置。useSelector ,reselect 都有,具体看文档。

    另外性能差异有多少,要实际测试才知道。
    sillydaddy
        8
    sillydaddy  
    OP
       2022-08-18 13:20:55 +08:00
    @kongkx
    啊,这个“compare 函数”提醒了我
    可以自定义 equality 方法,在比较参数是否相等的时候,把 store 类型的数据给忽略掉。虽然取巧,但感觉确实能达到目的。
    再次感谢,虽然没太弄懂你前面说的 key tree 方法。
    0xffSol
        9
    0xffSol  
       2022-08-18 18:47:45 +08:00 via iPhone
    可以使用 useContent 、useReducer 代替 redux
    ChrisV5
        10
    ChrisV5  
       2022-09-28 19:02:04 +08:00
    一般是 select parent as chilends 之后,再 chiledns.map(c=><div><RenderChildren/><div>)
    在 RenderChidren 里面再 Select
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     2773 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 27ms UTC 15:06 PVG 23:06 LAX 08:06 JFK 11:06
    Do have faith in what you're doing.
    ubao snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86