文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

原生React怎么实现懒加载列表

2023-07-05 16:36

关注

这篇文章主要介绍了原生React怎么实现懒加载列表的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇原生React怎么实现懒加载列表文章都会有所收获,下面我们一起来看看吧。

应用场景

懒加载列表或叫做无限滚动列表,也是一种性能优化的方式,其可疑不必一次性请求所有数据,可以看做是分页的另一种实现形式,较多适用于移动端提升用户体验,新闻、资讯浏览等。

效果预览

原生React怎么实现懒加载列表

思路剖析

container.clientHeight - wrapper.scrollTop <= wrapper.clientHeight

原生React怎么实现懒加载列表

原生代码实现

index.html

<body>  <div id="wrapper" onscroll="handleScroll()">    <ul id="container"></ul>  </div>  <script type="text/javascript" src="./index.js"></script></body>

index.css

* {  margin: 0;  padding: 0;}#wrapper {  margin: 100px auto;  width: 300px;  height: 300px;  border: 1px solid rgba(100, 100, 100, 0.2);  overflow-y: scroll;}ul#container {  list-style: none;  padding: 0;  width: 100%;}ul#container > li {  height: 30px;  width: 100%;}ul#container > li.green-item {  background-color: #c5e3ff;}ul#container > li.red-item {  background-color: #fff5d5;}

index.js

// 模拟数据构造const arr = [];const nameArr = ['Alice', 'July', 'Roman', 'David', 'Sara', 'Lisa', 'Mike'];let curPage = 1;let noData = false;const curPageSize = 20;const getPageData = (page, pageSize) => {  if (page > 5) return [];  const arr = [];  // const nameArr = ['Alice', 'July', 'Roman', 'David', 'Sara', 'Lisa', 'Mike'];  for (let i = 0; i < pageSize; i++) {    arr.push({      number: i + (page - 1) * pageSize,      name: `${nameArr[i % nameArr.length]}`,    });  }  return arr;};const wrapper = document.getElementById('wrapper');const container = document.getElementById('container');let plainWrapper = null;const handleScroll = () => {  // 当临界元素进入可视范围时,加载下一页数据  if (    !noData &&    container.clientHeight - wrapper.scrollTop <= wrapper.clientHeight  ) {    curPage++;    console.log(curPage);    const newData = getPageData(curPage, curPageSize);    renderList(newData);  }};const renderList = (data) => {  // 没有更多数据时  if (!data.length) {    noData = true;    plainWrapper.innerText = 'no more data...';    return;  }  plainWrapper && container.removeChild(plainWrapper); //移除上一个临界元素  const fragment = document.createDocumentFragment();  data.forEach((item) => {    const li = document.createElement('li');    li.className = item.number % 2 === 0 ? 'green-item' : 'red-item'; //奇偶行元素不同色    const text = document.createTextNode(      `${`${item.number}`.padStart(7, '0')}-${item.name}`    );    li.appendChild(text);    fragment.appendChild(li);  });  const plainNode = document.createElement('li');  const text = document.createTextNode('scroll to load more...');  plainNode.appendChild(text);  plainWrapper = plainNode;  fragment.appendChild(plainNode); //添加新的临界元素  container.appendChild(fragment);};// 初始渲染renderList(getPageData(curPage, curPageSize));

迁移到React

React 中实现时可以省去复杂的手动渲染逻辑部分,更关注数据。

store/data.ts

import { IDataItem } from '../interface';const nameArr = ['Alice', 'July', 'Roman', 'David', 'Sara', 'Lisa', 'Mike'];export const getPageData = (  page: number = 1,  pageSize: number = 10): Array<IDataItem> => {  if (page > 5) return [];  const arr = [];  // const nameArr = ['Alice', 'July', 'Roman', 'David', 'Sara', 'Lisa', 'Mike'];  for (let i = 0; i < pageSize; i++) {    arr.push({      number: i + (page - 1) * pageSize,      name: `${nameArr[i % nameArr.length]}`,    });  }  return arr;};

LazyList.tsx

import React, { FC, useCallback, useEffect, useReducer, useRef } from 'react';import { getPageData } from './store/data';import { IDataItem } from './interface';import styles from './index.module.css';export interface IProps {  curPageSize?: number;}export interface IState {  curPage: number;  noData: boolean;  listData: Array<IDataItem>;}const LazyList: FC<IProps> = ({ curPageSize = 10 }: IProps) => {  const clientRef: any = useRef(null);  const scrollRef: any = useRef(null);  const [state, dispatch] = useReducer(    (state: IState, action: any): IState => {      switch (action.type) {        case 'APPEND':          return {            ...state,            listData: [...state.listData, ...action.payload.listData],          };        default:          return { ...state, ...action.payload };      }    },    {      curPage: 1,      noData: false,      listData: [],    }  );    const handleScroll = useCallback(() => {    const { clientHeight: wrapperHeight } = scrollRef.current;    const { scrollTop, clientHeight } = clientRef.current;    // 当临界元素进入可视范围时,加载下一页数据    if (!state.noData && wrapperHeight - scrollTop <= clientHeight) {      console.log(state.curPage);      const newData = getPageData(state.curPage, curPageSize);      dispatch({        type: 'APPEND',        payload: { listData: newData },      });      dispatch({        payload: {          curPage: state.curPage + 1,          noData: !(newData.length > 0),        },      });    }  }, [state.curPage, state.noData]);  useEffect(() => {    const newData = getPageData(1, curPageSize);    dispatch({      type: 'APPEND',      payload: { listData: newData },    });    dispatch({      payload: {        curPage: 2,        noData: !(newData.length > 0),      },    });  }, []);  return (    <div className={styles[`wrapper`]} ref={clientRef} onScroll={handleScroll}>      <ul className={styles[`container`]} ref={scrollRef}>        {state.listData.map(({ number, name }) => (          <li            key={number}            className={              number % 2 === 0 ? styles[`green-item`] : styles[`red-item`]            }          >            {number}-{name}          </li>        ))}        {<li>{state.noData ? 'no more' : 'scroll'}</li>}      </ul>    </div>  );};export default LazyList;

关于“原生React怎么实现懒加载列表”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“原生React怎么实现懒加载列表”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯