0

0

如何防止滚动时重复触发页面渲染?

心靈之曲

心靈之曲

发布时间:2026-01-08 23:04:02

|

764人浏览过

|

来源于php中文网

原创

如何防止滚动时重复触发页面渲染?

本文详解如何通过正确管理事件监听器和状态更新,避免 react 组件因滚动事件频繁重渲染,尤其解决 `scrollposition` 变化导致动画重置的问题。核心在于仅在组件挂载时绑定一次滚动监听器,并确保 `useeffect` 不因无关依赖(如 `scrollposition`)重复执行。

你的代码存在两个关键问题:

  1. useEffect 依赖了 scrollPosition → 每次 setScrollPosition 触发后都会重新运行 animator(),导致重复添加监听器;
  2. animator 函数内部逻辑混乱:它试图用 animating 控制监听器,但该状态未被正确同步或用于防重绑定,且清理函数未被可靠执行。

✅ 正确做法是:将滚动监听器的绑定/解绑逻辑完全交由 useEffect 管理,且只在组件挂载/卸载时执行一次 —— 无需依赖 scrollPosition,更不需要 useMemo(原答案推荐的 useMemo 并不适用于此场景,它不能替代副作用管理)。

以下是优化后的完整实现:

Kubit.ai
Kubit.ai

一个AI驱动的产品分析平台,为产品和数据团队构建

下载
import { useState, useEffect } from 'react';

function App() {
  const [scrollPosition, setScrollPosition] = useState(0);

  useEffect(() => {
    const handleScroll = () => {
      setScrollPosition(window.scrollY);
    };

    // ✅ 仅在挂载时添加监听器,被动模式提升性能
    window.addEventListener('scroll', handleScroll, { passive: true });

    // ✅ 仅在卸载时移除监听器(cleanup 函数)
    return () => {
      window.removeEventListener('scroll', handleScroll);
    };
  }, []); // ? 空依赖数组:只执行一次

  // ✅ scrollPosition 现在仅反映真实滚动位置,不会意外触发重渲染链
  // 后续动画组件可基于该值做一次性初始化(例如配合 useRef + useEffect 判断首次变化)

  return (
    

当前滚动位置: {scrollPosition}px

{/* 示例:动画组件可在此处使用 scrollPosition 做条件触发 */}
); } // 示例动画组件:仅在滚动超过阈值时触发一次入场动画 function AnimatedSection({ scrollY }: { scrollY: number }) { const [isAnimated, setIsAnimated] = useState(false); const hasAnimatedRef = React.useRef(false); React.useEffect(() => { if (!hasAnimatedRef.current && scrollY > 100) { setIsAnimated(true); hasAnimatedRef.current = true; } }, [scrollY]); return (
✨ 这个区块仅在首次滚动过 100px 后动画进入
); }

? 关键要点总结

  • 永远不要把 scroll 监听器放在依赖 scrollPosition 的 useEffect 中 —— 它会导致无限循环或监听器堆积;
  • 使用 [] 作为 useEffect 依赖数组,确保监听器生命周期与组件严格对齐;
  • 对于“仅执行一次”的动画逻辑,用 useRef 标记状态(如 hasAnimatedRef),而非依赖 useState 触发重渲染;
  • 始终启用 { passive: true }(现代浏览器默认支持),避免滚动卡顿;
  • 若需节流/防抖滚动更新(如高频读取 scrollY),可结合 requestAnimationFrame 或 lodash.throttle,但本例中 setScrollPosition 本身已足够轻量。

这样修改后,scrollPosition 仍会实时更新,但组件不会因滚动反复重渲染,动画组件也能稳定保持状态,彻底解决“每次滚动都重置动画”的问题。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

61

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

31

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

73

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

20

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

24

2026.01.13

PHP缓存策略教程大全
PHP缓存策略教程大全

本专题整合了PHP缓存相关教程,阅读专题下面的文章了解更多详细内容。

7

2026.01.13

jQuery 正则表达式相关教程
jQuery 正则表达式相关教程

本专题整合了jQuery正则表达式相关教程大全,阅读专题下面的文章了解更多详细内容。

4

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.6万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号