【前端实战】如何让用户回到上次阅读的位置?

【前端实战】如何让用户回到上次阅读的位置?

目录

【前端实战】如何让用户回到上次阅读的位置?

一、总体思路

1、核心目标

2、涉及到的技术

二、实现方案详解

1、基础方法:监听滚动,记录 scrollTop(不推荐)

2、Intersection Observer + 插入探针元素

3、基于 URL Hash 锚点跳转

三、总结

1、不同方案间对比总结

2、结语


        作者:watermelo37

        ZEEKLOG万粉博主、华为云云享专家、阿里云专家博主、腾讯云、支付宝合作作者,全平台博客昵称watermelo37。

        一个假装是giser的coder,做不只专注于业务逻辑的前端工程师,Java、Docker、Python、LLM均有涉猎。



---------------------------------------------------------------------

温柔地对待温柔的人,包容的三观就是最大的温柔。

---------------------------------------------------------------------

【前端实战】如何让用户回到上次阅读的位置?

        在阅读类、资讯类、博客类网站中,记忆用户上次阅读到的位置,并在下次访问时自动滚动回那个位置,可以大大提升用户体验感。

        今天我们就来详细讲一讲:前端如何实现用户回到上次阅读的位置,包括基础scroll方法+优化、 Intersection Observer API + 探针追踪、锚点 URL HASH 定位跳转等策略实现一个流畅且高效的方案。

一、总体思路

1、核心目标

        在用户滚动时记录当前位置

        在页面重新加载时恢复到记录的位置

2、涉及到的技术

        可以根据下列技术进行知识补充,或者直接选择自己已经熟悉的技术来实现:

  • scroll 事件监听
  • localStorage 本地存储
  • requestAnimationFrame 节流优化
  • Intersection Observer API 观察元素进入视口
  • Vue3 响应式组件(如果需要框架版本)

二、实现方案详解

1、基础方法:监听滚动,记录 scrollTop(不推荐)

        在用户滚动时,实时记录 window.scrollY(页面滚动的垂直距离),保存到 localStorage 中。页面加载时,从 localStorage 读取,并 scrollTo 恢复。

        这可能是很多人的第一直觉,但是这种方法存在一个问题,就是scroll事件触发太频繁了。高频滚动下每秒触发一百多次都是非常正常的情况,常规的节流方法也不合适,比如添加一个节流时间。因为滚动可以是一个非常快速的过程,一秒钟可能可以操作滚动条从顶到尾。所以这里选择使用 requestAnimationFrame 方法来节流。

// 用于保存最新滚动位置 let lastKnownScrollY = 0; // 用于控制 requestAnimationFrame let ticking = false; // 监听滚动事件 window.addEventListener('scroll', () => { lastKnownScrollY = window.scrollY; // 防止过度频繁存储,使用requestAnimationFrame节流 if (!ticking) { window.requestAnimationFrame(() => { // 将滚动位置保存在localStorage中 localStorage.setItem('scrollPosition', lastKnownScrollY); ticking = false; }); ticking = true; } }); // 页面加载时,恢复之前保存的位置 window.addEventListener('DOMContentLoaded', () => { const savedPosition = localStorage.getItem('scrollPosition'); if (savedPosition !== null) { window.scrollTo(0, parseInt(savedPosition)); } }); 
        requestAnimationFrame 是浏览器提供的用于执行高效动画的 API,它会在下一次重绘前调用指定的回调函数,确保动画与屏幕刷新率同步(通常为 60Hz),从而实现平滑、流畅的视觉效果,同时避免不必要的性能开销。

        它的执行频率比 scroll 要小一些,同时不滚动的时候也不会触发,相比与 scroll 的高频触发,起到了一个节流效果。

2、Intersection Observer + 插入探针元素

        Intersection Observer在确定页面位置的时候有奇效,效率比scroll事件监听高了不止一星半点,但是如果存在大块、不宜分割或者杂乱的元素,那么监听元素的选择就会成为一个问题。

        添加探针元素可以有效解决这个问题,探针元素只需要小小一个 div,可以设置为 visibility: hidden,不影响页面布局。他们就像一个个哨兵,负责观察你的视口到了什么位置。

        (1)页面插入探针元素

        可以在重要段落、章节、标题前插入隐形的小 div。

<article> <div></div> <h2>第一章 标题</h2> <p>正文内容...</p> <div></div> <h2>第二章 标题</h2> <p>正文内容...</p> <!-- 更多内容 --> </article> 

        (2)设置 Intersection Observer,并在页面加载时,滚动到探针位置

// 创建 IntersectionObserver 实例 const observer = new IntersectionObserver((entries) => { entries.forEach(entry => { if (entry.isIntersecting) { // 如果探针元素进入可视区,记录它的id localStorage.setItem('lastVisibleSectionId', entry.target.id); } }); }, { threshold: 0.5 // 元素至少50%可见时触发 }); // 监听所有探针元素 document.querySelectorAll('.observer-marker').forEach(marker => { observer.observe(marker); }); // 页面加载时,恢复到上次记录的探针 window.addEventListener('DOMContentLoaded', () => { const lastId = localStorage.getItem('lastVisibleSectionId'); if (lastId) { const element = document.getElementById(lastId); if (element) { element.scrollIntoView({ behavior: 'smooth' }); // 平滑滚动到探针 } } }); 

3、基于 URL Hash 锚点跳转

        给每一节内容设置唯一 id,用户阅读到某个位置时,自动更新 URL 的 hash(锚点 #id),页面加载时,浏览器根据 hash 自动滚动到对应位置。这种方式实现的跳转甚至可以实现分享,因为位置信息是保存在 URL 里面的。

        是不是很熟悉?ZEEKLOG的目录跳转就是这么实现的。

// 监听页面滚动,动态更新 URL Hash const observer = new IntersectionObserver((entries) => { entries.forEach(entry => { if (entry.isIntersecting) { // 动态替换地址栏 hash,不刷新页面 history.replaceState(null, '', `#${entry.target.id}`); } }); }, { threshold: 0.5 }); // 监听所有需要作为锚点的元素 document.querySelectorAll('.observer-marker').forEach(marker => { observer.observe(marker); }); // 页面刷新后,浏览器会自动滚动到hash对应的元素 

三、总结

1、不同方案间对比总结

方法优点缺点适用场景

scrollTop 记录

通用、简单粗糙、动态内容页面误差大小型项目、静态页面
Intersection Observer 探针精准、性能好要布置探针,稍复杂长内容、章节型页面
URL Hash 锚点轻便、天然支持浏览器跳转地址栏变化,需考虑SEO文章分享、文档导航

2、结语

        实现回到上次阅读位置,并不只有一种方式,关键是根据你的项目特点选择:

  • 内容简单 ➔ scrollTop 就够了。
  • 内容结构清晰 ➔ Intersection Observer是最佳。
  • 需要分享/跳转 ➔ 用 URL Hash 最自然。

        总之,真正优秀的细节体验,源自对用户行为的深刻理解和用心打磨。

        只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~

        其他热门文章,请关注:

        极致的灵活度满足工程美学:用Vue Flow绘制一个完美流程图

        你真的会使用Vue3的onMounted钩子函数吗?Vue3中onMounted的用法详解

        DeepSeek:全栈开发者视角下的AI革命者

        通过array.filter()实现数组的数据筛选、数据清洗和链式调用

        通过Array.sort() 实现多字段排序、排序稳定性、随机排序洗牌算法、优化排序性能

        TreeSize:免费的磁盘清理与管理神器,解决C盘爆满的燃眉之急

        通过MongoDB Atlas 实现语义搜索与 RAG——迈向AI的搜索机制

        深入理解 JavaScript 中的 Array.find() 方法:原理、性能优势与实用案例详解

        el-table实现动态数据的实时排序,一篇文章讲清楚elementui的表格排序功能

        前端实战:基于Vue3与免费满血版DeepSeek实现无限滚动+懒加载+瀑布流模块及优化策略

        MutationObserver详解+案例——深入理解 JavaScript 中的 MutationObserver

        JavaScript中通过array.map()实现数据转换、创建派生数组、异步数据流处理、DOM操作等

        高效工作流:用Mermaid绘制你的专属流程图;如何在Vue3中导入mermaid绘制流程图

        干货含源码!如何用Java后端操作Docker(命令行篇)

        在线编程实现!如何在Java后端通过DockerClient操作Docker生成python环境

        Dockerfile全面指南:从基础到进阶,掌握容器化构建的核心工具

Read more

uni-app——uni-app 小程序 之 【按钮失效问题排查(前端+后端)】

一、问题背景 在某业务流程系统中,当业务单据进入特定待处理状态后,用户需要在对应操作页面完成核心操作,点击页面中的两个关键操作按钮(提交类、完结类)以推进流程流转。 然而实际操作时,两个按钮均出现报错提示,无法正常触发流程跳转,业务无法继续推进。 二、问题复现 操作步骤: 1. 登录系统账号(具备对应操作权限) 2. 进入业务单据列表,找到一条处于特定待处理状态的单据 3. 点击进入该单据的操作详情页面 4. 填写页面所需基础信息、上传相关附件 5. 点击页面中的提交类或完结类按钮 6. 结果:按钮点击后报错,流程无法流转到下一节点,操作失败。 三、问题分析 经过多轮排查,发现问题并非单一环节导致,而是涉及前端和后端两层,属于接口调用、参数传递及数据校验的联动异常,具体分析如下: 1. 前端:页面加载时未获取核心业务数据 操作详情页面进入后,未先调用查询接口获取单据关联的核心数据,直接使用空值的关键标识调用操作接口,导致后端无法查询到对应业务记录,接口调用失败。

前端代码质量保证:让你的代码更可靠

前端代码质量保证:让你的代码更可靠 毒舌时刻 代码质量?听起来就像是前端工程师为了显得自己很专业而特意搞的一套复杂流程。你以为随便写几个测试就能保证代码质量?别做梦了!到时候你会发现,测试代码比业务代码还多,维护起来比业务代码还麻烦。 你以为ESLint能解决所有问题?别天真了!ESLint只能检查代码风格,无法检查逻辑错误。还有那些所谓的代码质量工具,看起来高大上,用起来却各种问题。 为什么你需要这个 1. 减少错误:代码质量保证可以帮助你发现和修复代码中的错误,减少生产环境中的问题。 2. 提高可维护性:高质量的代码更容易理解和维护,减少后期的维护成本。 3. 促进团队协作:统一的代码质量标准可以便于团队成员之间的协作,减少沟通成本。 4. 提高开发效率:高质量的代码可以减少调试和修复错误的时间,提高开发效率。 5. 提升代码安全性:代码质量保证可以帮助你发现和修复安全漏洞,提升代码的安全性。 反面教材 // 这是一个典型的代码质量问题示例 // 1. 代码风格不一致 function getUser(id) { return fetch(`/api/

AI大模型驱动的软件开发全流程变革:从需求分析到智能运维的技术演进与未来展望

AI大模型驱动的软件开发全流程变革:从需求分析到智能运维的技术演进与未来展望

目录 * 引言:当软件开发遇上"工业革命4.0" * 一、需求分析:从用户故事到智能需求工程 * 1.1 智能需求解析器 * 1.2 需求验证闭环 * 二、设计阶段:AI架构师的诞生 * 2.1 微服务自动设计 * 2.2 技术选型决策树 * 三、编码阶段:从辅助到主导 * 3.1 多语言代码生成 * 3.2 代码审查革命 * 四、测试阶段:质量保证的范式转移 * 4.1 智能测试用例生成 * 4.2 缺陷预测模型 * 五、部署与运维:自愈式系统的崛起 * 5.1 智能容量规划 * 5.

数字人技术的核心:AI与动作捕捉的双引擎驱动(2/10)

数字人技术的核心:AI与动作捕捉的双引擎驱动(2/10)

摘要:数字人技术从静态建模迈向动态交互,AI与动作捕捉技术的深度融合推动其智能化发展。尽管面临表情僵硬、动作脱节、交互机械等技术瓶颈,但通过多模态融合技术、轻量化动捕方案等创新,数字人正逐步实现自然交互与情感表达。未来,数字人将成为连接物理世界与数字空间的虚拟生命体,推动社会进入虚实共生的新纪元。 一、数字人技术的演进与核心瓶颈 1.1 从静态建模到动态交互的跨越 数字人技术的发展历程,是一部从简单到复杂、从静态呈现到动态交互的技术进化史。早期,数字人主要以静态建模的形式出现在影视、游戏等领域,依靠手工细致地构建模型与精心预设的动作库,来赋予数字人基本的形象和动作。但这种方式下的数字人,宛如被定格在特定场景中的木偶,缺乏与外界实时互动的能力,也难以展现出丰富多元的个性化表达。 随着元宇宙概念如风暴般席卷而来,虚拟世界与现实世界的边界变得愈发模糊,用户对于虚拟形象的要求达到了前所未有的高度。他们渴望数字人不再是机械、刻板的存在,而是能如同真实人类一般,自然流畅地交流,根据不同情境做出恰当反应,拥有独特的个性魅力 。这种强烈的需求,如同汹涌的浪潮,推动着数字人技术朝着智能化、实时