关于react中列表渲染的局部刷新问题

 更新时间:2022年08月04日 14:34:10   作者:ConanZzz_  
这篇文章主要介绍了关于react中列表渲染的局部刷新问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

react中列表渲染的局部刷新

最近在写demo的时候遇到一个更新列表中某个的对象的某个值,最期待的结果肯定是局部刷新,但是我们往往在改变值之后会遇到全局都刷新的问题,以下为个人实验出来的一个小技巧。

首先我有以下数据需要通过react的列表方法渲染:

let list=[
    {
        id:1,
        show:false
    },
    {
        id:2,
        show:false
    },
    {
        id:3,
        show:false
    }
]

我们通过以下react方法进行渲染:

render(){
    return (
        {list.map((val)=>{
            <DemoComponent val={val}/>
        })}
    )
}

在这里我们需要重新写一个DemoComponent的组件:

import React,{Component} from 'react';
export class DemoComponent extends Component{
    render(){
        <div>
          <div>this.props.val.id</div>
          <button onClick={()=>this.toggleDialog()}>toggle</button>  
        </div>
    }
    toggleDialog(){
        // 更改val.showDialog相关操作。
    }
    shouldComponentUpdate(nextProps){
        return JSON.stringify(nextProps) !== JSON.stringify(this.props);
    }
}

当我们点击第一个button,这个时候就能达到局部刷新,只刷新第一个DemoComponent组件的效果了。

但以上操作是什么原理呢?

首先,大家平时都推荐使用的PureComponent不能在这里使用,因为这个组件没有shouldComponentUpdate这个钩子函数,虽然PureComponent也有对比props和nextProps并自行判断当前组件是否需要重新渲染的功能,但是这个对比对对象是没有用的,因为{} === {}是返回false的(对这个知识点不理解的朋友可以去看看堆栈相关的知识),数组同理。

那么为了让其他列表组件没有必要多去render一次,所以我在shouldComponentUpdate中取了个巧,直接JSON.stringify,将两个对象转换成字符串进行比较,这样就方便的多。

当然这里也是自己为了不去写一个isEqual方法而偷懒的做法,这样做的好处是简单方便,也节省时间(isEqual方法必然会for-in循环,对于更复杂的情况甚至需要递归,在内存消耗和时间复杂度上肯定会比JSON.stringify严重),缺点在于可拓展性不高,如果是个数组,就比较头疼。 

react实现实时/局部刷新

在React项目中,经常会遇到诸如:删除某一行的同时同时刷新表格展示最新的表格,而不是删除后,手动刷新。

原先是在删除时再次调用生命周期函数中展示表格列表的函数,但会影响性能。

总结:通过JS数组修改usestate中的状态。

初始做法示例

useEffect(() => {
        httpOffer()
    }, [])
//获取列表
    const httpOffer = () => {
        offerList()
            .then(res => {
                // console.log('offer', res)
                if (res.status === 200) {
                    if (res.data.code === 0) {
                        const data = res.data.data
                        data.length > 0 && data.forEach(el => {
                            el.key = el.id
                        });
                        setOffers(data)
                    } else {
                        message.error(res.data.msg)
                    }
                } else {
                    message.info('network failed')
                }
                setLoading(false)
            })
            .catch(err => {
                setLoading(false)
                throw err
            })
    }
const showModalDel = (id) => {
        delOffer({ id })
            .then(res => {
                if (res.status === 200) {
                    if (res.data.code === 0) {
                        //之前的做法,刷新页面
                        httpOffer()
                        message.info(res.data.msg)
                    } else {
                        message.error(res.data.msg)
                    }
                } else {
                    message.error('Network failed!')
                }
            })
            .catch(err => {
                throw err
            })
    }

目前可以使用JS数组和usestate进行同样的操作。

增加数据

addOffer(params)
                .then(res => {
                    if (res.status === 200) {
                        if (res.data.code === 0) {
                            let data = res.data.data
                            //重点
                            setOffers([...offers, { key: data.id, id: data.id, ...params}])
                            
                            message.info(res.data.msg)
                            setVisible(false)
                        } else {
                            message.error(res.data.msg)
                        }
                    } else {
                        message.error('Network failed!')
                    }
                })
                .catch(err => {
                    console.log(err)
                })

修改数据

editOffer(params)
                .then(res => {
                            //重点
                            setOffers(offers.map(item => {
                                if(item.id === params.id){
                                    item.id = params.id,
                                    item.adv_id = params.adv_id,
                                }
                                return item
                            }))
                            setVisible(false)
                })
                .catch(err => {
                    console.log(err)
                })

删除数据

const showModalDel = (id) => {
        delOffer({ id })
            .then(res => {
                 //重点
                 setOffers(offers.filter(item => item.id != id ))
                 message.info(res.data.msg)
            })
            .catch(err => {
                throw err
            })
    }

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。 

相关文章

  • React项目使用ES6解决方案及JSX使用示例详解

    React项目使用ES6解决方案及JSX使用示例详解

    这篇文章主要为大家介绍了React项目使用ES6解决方案及JSX使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • React前端渲染优化--父组件导致子组件重复渲染的问题

    React前端渲染优化--父组件导致子组件重复渲染的问题

    本篇文章是针对父组件导致子组件重复渲染的优化方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • React redux 原理及使用详解

    React redux 原理及使用详解

    这篇文章主要为大家介绍了React redux 原理及使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • React 事件绑定的实现及区别

    React 事件绑定的实现及区别

    事件绑定也是其中一部分内容,通过事件委托和事件合成,React 在内部对事件进行优化和处理,减少了事件处理函数的调用次数,从而提升了性能,本文主要介绍了React事件绑定的实现及区别,感兴趣的可以了解一下
    2024-03-03
  • React18新特性startTransition详解

    React18新特性startTransition详解

    React18的新特性startTransition主要是为了优化用户体验,通过标记低优先级的更新任务,如页面重渲染,使它们不会阻塞高优先级的紧急任务如用户输入响应,本文介绍React18新特性startTransition,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • react native实现监控手势上下拉动效果

    react native实现监控手势上下拉动效果

    这篇文章主要为大家详细介绍了react native实现监控手势上下拉动效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-05-05
  • 使用react-virtualized实现图片动态高度长列表的问题

    使用react-virtualized实现图片动态高度长列表的问题

    一般我们在写react项目中,同时渲染很多dom节点,会造成页面卡顿, 空白的情况。为了解决这个问题,今天小编给大家分享一篇教程关于react-virtualized实现图片动态高度长列表的问题,感兴趣的朋友跟随小编一起看看吧
    2021-05-05
  • react中如何使用监听

    react中如何使用监听

    在 React 中,您可以使用 addEventListener 函数来监听事件,本文通过实例代码给大家介绍react中如何使用监听,感兴趣的朋友跟随小编一起看看吧
    2023-10-10
  • React文字展开收起组件的实现示例

    React文字展开收起组件的实现示例

    本文主要介绍了React文字展开收起组件的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • React日期时间显示组件的封装方法

    React日期时间显示组件的封装方法

    这篇文章主要为大家详细介绍了React日期时间显示组件的封装方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08

最新评论