react hooks setData设置数据 一直被替换掉 无法追加数据

经过后续的查找资料,发现在react hooks中setState 之所以会拿到上一次的缓存,是因为,函数在每次渲染时也是独立的(即在内存中开辟不同的空间)。这就是 Capture Value 特性。

解决办法除了以下方法之外,还有3种:https://blog.csdn.net/hzxOnlineOk/article/details/109103135

博主要实现一个分页追加数据的功能

  let [childOrgList, setChildOrgList] = useState([]);
const initData = _.throttle(() => {
    const option = { pid: location.state && location.state.orgId, category: location.state && location.state.category, pageNum: pageNum, pageSize: PAGESIZE };
    dispatch({ type: '$/_getChildOrg', payload: option }).then((r) => {
      if (r) {
        isLoading = false;
        const data = (r && r.list) || [];

        console.log('childOrgList===>', childOrgList, data)
        if (aniLoading) {
          setAniLoading(false);
        }
        if (data && data.length < PAGESIZE) {
          // 没有更多了
          if (!isNoMore) {
            setLoading(false);
            setChildOrgList([...childOrgList, ...data]);
          }
          setIsNoMore(true);
        } else {
          pageNum += 1;
          setLoading(false);
          setChildOrgList([...childOrgList, ...data]);
        }
      } else {
        setAniLoading({ aniLoading: false })
      }
    });
  }, 1000)

 

但是数据一直无法追加上去!!!

经过2个小时的探索,发现,需要使用setChildOrgList(prevState => [...prevState, ...data])的形式,才能成功追加数据...

const initData = _.throttle(() => {
    const option = { pid: location.state && location.state.orgId, category: location.state && location.state.category, pageNum: pageNum, pageSize: PAGESIZE };
    dispatch({ type: '$/_getChildOrg', payload: option }).then((r) => {
      if (r) {
        isLoading = false;
        const data = (r && r.list) || [];

        console.log('childOrgList===>', childOrgList, data)
        if (aniLoading) {
          setAniLoading(false);
        }
        if (data && data.length < PAGESIZE) {
          // 没有更多了
          if (!isNoMore) {
            setLoading(false);
            setChildOrgList(childOrgList => [...childOrgList, ...data]);
          }
          setIsNoMore(true);
        } else {
          pageNum += 1;
          setLoading(false);
          setChildOrgList(childOrgList => [...childOrgList, ...data]);
        }
      } else {
        setAniLoading({ aniLoading: false })
      }
    });
  }, 1000)

完整代码 PartyOrgList.js

import React, { useEffect, useState } from 'react';
import PropTypes from 'prop-types';
import { connect } from 'dva';
import { Flex, List, SearchBar, WhiteSpace, } from 'antd-mobile';
import LoadWrap from '../../components/LoadWrap';
import NavBar from '../../components/navBar';
import styles from './partyOrgList.less';
import _ from 'lodash';
import $ from 'jquery';
import { avatarUrl } from '../../utils/helper';

const PartyOrgList = props => {
  let pageNum = 1;
  const PAGESIZE = 15;
  let height = $(document).height() - 90;

  let isLoading = false;

  let [pageStackIndex, setPageStackIndex] = useState(0);
  let [childOrgList, setChildOrgList] = useState([]);
  let [isNoMore, setIsNoMore] = useState(false);
  let [loading, setLoading] = useState(false); // 用于控制分页的loading
  let [aniLoading, setAniLoading] = useState(true); // 用于控制LoadWrap容器
  const { location, dispatch, history } = props;

  useEffect(() => {
    console.log('location, history', location, history)
    initData();
    $('.signin_list').scroll(() => {
      //已经滚动到上面的页面高度
      var scrollTop = $('.signin_list').scrollTop();
      // 列表容器高度
      var containerHeight = $('.page_list').height();
      //浏览器窗口高度
      var windowHeight = $(window).height();
      console.log('scrollTop, height, containerHeight', scrollTop, height, containerHeight)
      //此处是滚动条到底部时候触发的事件,在这里写要加载的数据,或者是拉动滚动条的操作
      if (scrollTop + height >= containerHeight - 20) { // 1页的高度 总高度 滚动高度
        setLoading(true);
        if (isNoMore || isLoading === true) {
          return;
        }
        isLoading = true;
        initData();
      }
    });
  }, [location.state && location.state.category]) // 只传[] 等同于componentDidMount的效果

  const initData = _.throttle(() => {
    const option = { pid: location.state && location.state.orgId, category: location.state && location.state.category, pageNum: pageNum, pageSize: PAGESIZE };
    dispatch({ type: '$/_getChildOrg', payload: option }).then((r) => {
      if (r) {
        isLoading = false;
        const data = (r && r.list) || [];

        console.log('childOrgList===>', childOrgList, data)
        if (aniLoading) {
          setAniLoading(false);
        }
        if (data && data.length < PAGESIZE) {
          // 没有更多了
          if (!isNoMore) {
            setLoading(false);
            setChildOrgList(childOrgList => [...childOrgList, ...data]);
          }
          setIsNoMore(true);
        } else {
          pageNum += 1;
          setLoading(false);
          setChildOrgList(childOrgList => [...childOrgList, ...data]);
        }
      } else {
        setAniLoading({ aniLoading: false })
      }
    });
  }, 1000)

  // 根据不同页面设置navbar
  function renderNavBar() {
    return (
      <div className={styles.navbarWrap}>
        <NavBar title={<div className={styles.navTitle}>组织列表</div>} onLeftClick={() => props.history.replace('/')} />
      </div>
    );
  }

  function renderInit() {
    // console.log('data childOrgList', childOrgList, aniLoading)

    return (
      <Flex direction="column" align="stretch" justify="between" className="flex100" style={{ background: '#f6f6f6' }}>
        <WhiteSpace style={{ background: '#f6f6f6', height: 10 }} />
        <div className="signin_list" style={{ overflow: 'auto', height: height }}>
          <div className="page_list">
            {
              !aniLoading ? <LoadWrap data={{ data: childOrgList }} emptyString="暂无数据">
                {(childOrgList && childOrgList.length && childOrgList || []).map((item, index) => {
                  const src = avatarUrl(item.oaCode);
                  const short = item.name && item.name.substring(0, 1);

                  return (
                    <List key={index} style={{ marginBottom: 8, border: 'none' }}>
                      <List.Item arrow="horizontal">
                        <div style={{ display: 'flex', flexDirection: 'row' }}>
                          <div style={{ display: 'flex', flexDirection: 'column' }}>
                            <span style={{
                              fontSize: 16, color: '#333', lineHeight: '20px',
                              // overflow: 'hidden',
                              // textOverflow: 'ellipsis',
                              // display: '-webkit-box',
                              // webkitLineClamp: '2',
                              // webkitBoxOrient: 'vertical',
                            }}>{item}</span>
                          </div>
                        </div>
                      </List.Item>
                    </List>
                  )
                })}

                {loading && !isNoMore ? (
                  <div style={{ padding: '15px 0', fontSize: 14, color: '#999', display: 'flex', alignItems: 'center', justifyContent: 'center' }}>
                    <span className="am-activity-indicator-spinner" style={{ width: 15, height: 15 }} />
                    <span style={{ marginLeft: 5, }}>正在加载</span>
                  </div>) : null
                }

                {isNoMore && childOrgList && childOrgList.length > PAGESIZE ? <div style={{ padding: '15px 0', fontSize: 14, color: '#999', textAlign: 'center' }}>没有更多了</div> : null}
              </LoadWrap> : <div className={styles.spinner} style={{ position: 'relative' }} />
            }
          </div>
        </div>
      </Flex>
    );
  }

  return (
    <Flex
      direction="column"
      align="stretch"
      className="flex100"
      style={{ backgroundColor: '#F6F6F6' }}
    >
      {renderNavBar()}
      <Flex.Item
        className={`flex100 ${styles.content}`}
        style={{ height: 'auto', overflow: 'hidden' }}
      >
        {renderInit()}
      </Flex.Item>
    </Flex>
  );
}

function mapStateToProps(state, ownProps) {
  // const data = state.signIn;
  return {
    // childOrgList: data.childOrgList,
  };
}

export default connect(mapStateToProps)(PartyOrgList);

partyOrgList.less 

.signWrap {
  height: 70px;
  padding: 0 15px;
  background: #6ba3f1;
  display: flex;
  flex-direction: row;
  justify-content: center;
  align-items: center;
}

.spinner {
  margin: 5rem auto;
  width: 2rem;
  height: 2rem;
  position: relative;
  background: center center/100% 100% no-repeat
    url('');
}

.spinner::before {
  top: 0;
  left: 0;
  content: '';
  width: 100%;
  height: 100%;
  position: absolute;
  animation: rotates 1s linear infinite;
  background: center center/100% 100% no-repeat
    url('');
}

.avatar {
  width: 40px;
  height: 40px;
  display: inline-block;
  // margin-bottom: 5px;
  border-radius: 50%;
  background-color: #d8be85;
  color: #fff;
}

.avatarImg {
  width: 40px !important;
  height: 40px !important;
  border-radius: 50%;
}

.avatar em {
  font-style: normal;
  font-size: 15px;
  line-height: 41px;
  color: #fff !important;
}

.user {
  text-align: center;
  width: 55px;
}

.user .name {
  font-size: 12px;
  white-space: nowrap;
  text-overflow: ellipsis;
  overflow: hidden;
}

:global {
  .am-search {
    background: #fff;
    border-bottom: 1px solid #ddd;
  }
  .am-search-input {
    background: #f7f7f7;
  }
  .am-search-cancel {
    color: #999;
  }
}

@keyframes rotates {
  0% {
    transform: rotate(0deg);
  }
  100% {
    transform: rotate(360deg);
  }
}

@media screen and (max-width: 380px) {
  .navTitle {
    font-size: 16px;
  }
}

@media screen and (max-width: 350px) {
  .navTitle {
    font-size: 14px;
  }
}

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hzxOnlineOk

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值