nuxt3正确使用keepalive页面缓存组件缓存
最近使用nuxt@3.x版本做SEO优化项目比较多,之前也踩坑过,所以记录一下在 nuxt3 中路由缓存的正确使用方法,本人也之前在GitHub社区中提交过反馈问题,最后是在 3.8.2 版本解决了路由缓存问题。下面讲解如何正确使用keepalive做到页面缓存,组件缓存。
# 环境版本如下
node # 21.4.0
nuxt # 3.12.3
vue # latest 目前最新版本 3.4.*
页面缓存
keepalive 我们知道都是用来缓存组件 在组件卸载的时候 并不去真正意义上的销毁,而是隐藏掉。等再次挂载的时候再把它显示出来。其组件的状态保持原有的状态,并不会初始化。写多了 vue 项目的小伙伴们,大部分通过路由文件( src/router.js )定义 name 值,再去 router-view 组件里包裹 keepalive 组件去判断 路由里面的 name 值去使用。但是在 nuxt 框架内不需要这么复杂的操作。
- app.vue文件内容
<!-- app.vue -->
<template>
<div>
<!-- 最新版vue支持的语法,老版本可能提示错误 -->
<NuxtPage :keepalive />
</div>
</template>
<script lang="ts" setup>
// keepalive 所需的参数 指定name值为index 的页面 进行缓存
const keepalive = {
include: ["index"],
};
</script>
- pages文件夹
pages
├─index.vue # 若组件未定义name值 则为 文件名 index
└─user.vue # name为user
- index.vue文件内容
<template>
<div>index</div>
<p>data: {{ data }}</p>
<p>
<button @click="data++">+1</button>
<button @click="data--">-1</button>
</p>
<p class="links">
<NuxtLink to="/user">user Page</NuxtLink>
</p>
</template>
<script setup lang="ts">
const data = ref(0);
</script>
项目启动路径为 http://localhost:3000/ 的时候,此时,在组件内操纵 data 的值变化,再去跳转 http://localhost:3000/user 时,再跳回 http://localhost:3000/ , data 的值不会初始化,而是切换 /user 路由前页面的状态。此时使用 onActivated api来监听组件被激活。
<script lang="ts" setup>
onActivated(() => {
console.log("onActivated 页面激活了!");
});
</script>
- 路径复杂的请使用
defineOptions指定组件name值,以免使用路由缓存失败!
例如:pages/userData.vue,pages/news/detail,pages/news/[id].vue。这几个路径过于复杂,在 NuxtPage 组件中难以使用 include 属性值去判断缓存条件,所以需要在页面文件中声明该页面的组件 name 值。
<!-- pages/index.vue -->
<script setup lang="ts">
defineOptions({
name: "IndexPage",
});
</script>
<!-- pages/news/detail.vue -->
<script setup lang="ts">
defineOptions({
name: "newsDetail",
});
</script>
<!-- app.vue -->
<template>
<div>
<NuxtPage :keepalive />
</div>
</template>
<script lang="ts" setup>
const keepalive = {
include: ["IndexPage","newsDetail"],
// 对应 pages/index.vue ,pages/news/detail.vue 中的 name 值 缓存路由
};
</script>
组件缓存
如果在页面中,使用了v-if指令来控制组件显示,如何保证组件数据不被销毁。当然还是使用keepalive组件。假设我们在components文件中定义组件,会自动全局导入,无需引用。
components
├─Val
│ └─Input.vue # 若组件未定义name值 则为 文件名 Input
├─Counter.vue # 若组件未定义name值 则为 文件名 Counter
└─TextTip.vue # name 为 TextTip
<!-- components/Counter.vue -->
<template>
<div class="counter">
<h2>counter</h2>
<p>data:{{ data }}</p>
<p>
<button @click="data++">+1</button>
<button @click="data--">-1</button>
</p>
</div>
</template>
<script lang="ts" setup>
const data = ref(0);
</script>
<style scoped>
.counter {
padding: 20px;
}
</style>
<!-- components/TextTip.vue -->
<template>
<div class="text-tip">
<h2>text-tip</h2>
<p>text: {{ text }}</p>
<p>
<button @click="text += '1'">add Text</button>
</p>
</div>
</template>
<script lang="ts" setup>
const text = ref("text-1");
</script>
<!-- pages/index.vue -->
<template>
<div>index</div>
<p>
<button @click="showCounter = !showCounter">
{{ !showCounter ? "显示Counter" : "显示TextTip" }}
</button>
</p>
<hr />
<KeepAlive :include="keep.include">
<Counter v-if="showCounter" />
<TextTip v-else />
</KeepAlive>
</template>
<script setup lang="ts">
const showCounter = ref(false);
const keep = {
include: ["Counter"],
};
</script>
此时点击 显示Counter 按钮,在 Counter 组件中操作内部数据改变 data:5 ,点击按钮再去切换组件显示隐藏,会发现 Counter 组件并不会销毁掉之前的值 data:5 ,而TextTip组件在操作内部的数据改变后切换 隐藏/显示 后, text 数据是初始化的值 text-1 。
- 注意components文件定义的组件name值
components/Counter.vue , components/Val/Input.vue 的 name 值nuxt自动会给组件的name值取为文件名 Counter , Input 。而在组件自动导入的时候却是使用 <Counter /> , <ValInput /> 。会有点迷惑,所以请在使用 <KeepAlive> 组件包裹来缓存状态,请务必使用 defineOptions 指定组件的name值。
<!-- components/Val/Input.vue -->
<template>
<div class="input-warpper">
<h2>Input</h2>
<input v-model="val" />
</div>
</template>
<script lang="ts" setup>
const val = ref("");
defineOptions({
name: "ValInput"
})
</script>
<style>
.input-warpper {
margin: 20px;
}
</style>
或者使用 export default { name:'xxx' } 来指定组件的name值,不使用 setup 语法。
<!-- components/Val/Input.vue -->
<template>
<div class="input-warpper">
<h2>Input</h2>
<input v-model="val" />
</div>
</template>
<script lang="ts" setup>
const val = ref("");
</script>
<script lang="ts">
export default {
name: "ValInput"
}
</script>
<style>
.input-warpper {
margin: 20px;
}
</style>
案例地址
点击这里跳转代码案例,来调试 keepalive 页面缓存 和 组件缓存
推荐环境版本: node v21.4.0 , nuxt v3.12.* , 使用 pnpm 安装依赖。
nuxt3正确使用keepalive页面缓存组件缓存的更多相关文章
- Django学习之十二:Cache 缓存组件
目录 Django Cache 缓存组件 缓存逻辑伪代码 配置缓存源 可配置参数说明 01. Django的默认缓存 02. 基于Redis的django-redis 03. 自定义cache 04. ...
- Vue源码解析,keep-alive是如何实现缓存的?
前言 在性能优化上,最常见的手段就是缓存.对需要经常访问的资源进行缓存,减少请求或者是初始化的过程,从而降低时间或内存的消耗.Vue 为我们提供了缓存组件 keep-alive,它可用于路由级别或组件 ...
- Vue keep-alive如何实现只缓存部分页面
prop: include: 字符串或正则表达式.只有匹配的组件会被缓存. exclude: 字符串或正则表达式.任何匹配的组件都不会被缓存. 在2.1.0版本Vue中 常见用法: // 组件 exp ...
- vue 缓存的keepalive页面刷新数据
用到这个的业务场景是这样的: a页面点击新建列表按钮进入到新建的页面b,填写b页面并点击b页面确认添加按钮,把这些数据带到a页面,填充到列表(数组),可以添加多条, 点击这条的时候进入到编辑页面,确认 ...
- Vue匿名组件使用keep-alive后动态清除缓存
在使用Vue开发管理系统项目的时候,为了保存页面的浏览状态,我们可以使用内置组件keep-alive来缓存组件内部状态,避免重新渲染. <keep-alive> <router-vi ...
- vue-learning:34 - component - 内置组件 - 缓存组件keep-alive
vue内置缓存组件keep-alive <keep-alive>标签内包裹的组件切换时会缓存组件实例,而不是销毁它们.避免多次加载相应的组件,减少性能消耗.并且当组件在 <keep- ...
- vue的组件缓存(返回页面不刷新)
每次使用返回是页面总是会刷新 导致了一些体验上的不愉快 现在 发现vue中的一个很方便的方法还可以用来优化性能就是: keep-alive缓存组件 <router-view v-if=" ...
- React中实现keepalive组件缓存效果
背景:由于react官方并没有提供缓存组件相关的api(类似vue中的keepalive),在某些场景,会使得页面交互性变的很差,比如在有搜索条件的表格页面,点击某一条数据跳转到详情页面,再返回表格页 ...
- keep-alive vue组件缓存避免多次加载相应的组件
keep-alive vue组件缓存避免多次加载相应的组件
- vue 2.0 路由切换以及组件缓存源代码重点难点分析
摘要 关于vue 2.0源代码分析,已经有不少文档分析功能代码段比如watcher,history,vnode等,但没有一个是分析重点难点的,没有一个是分析大命题的,比如执行router.push之后 ...
随机推荐
- 4G EPS 第四代移动通信系统
目录 文章目录 目录 4G EPS 4G EPS 4G(the 4th generation mobile communication technology,第四代移动通信技术)提供了 3G 不能满足 ...
- 化繁为简|AIRIOT智慧水务信息化建设解决方案
"生产自动化,管理信息化"是现代化水厂建设的目标之一,需要在水质要求.工艺.生产.管理.环境等监测方面达到精细化管理标准,这是一个高度智能化,实现化繁为简智慧进阶的工程.传统水 ...
- 如何部署ASP.NET Core到Linux服务器
如何部署ASP.NET Core 到Linux服务器 我们开发的最终目的,是将开发后的东西发布网络上,以便自己及其他人使用. 本篇博客介绍如果在 linux 上部署 ASP.NET Core应用,使用 ...
- Avalonia中的线性渐变画刷LinearGradientBrush
在WPF中使用Shape实现复杂线条动画后,尝试在Avalonia中也实现同样效果.尽管官方提供了从WPF到Avalonia的快速入门文档,但由于第一次使用Avalonia,体验过程中并不是很顺利,主 ...
- C# Linq、Lambda表达式树动态构建、合并条件扩展方法
前言 日常开发时,使用Linq和EF经常会在存在多条件查询,或者说动态条件查询时,便存在合并表达式树的情况.基于这种情况结合一些资料,写了个扩展类,代码如下: 代码实现 /// <summary ...
- C# EF 使用sqlite 数据库出现表名出现dbo的坑
当ef使用sqlite时,正常情况映射的表名是没有dbo开头的.这个dbo是映射的sa用户,而sqlite是没有用户的.所以映射出的sql语句是查不到数据的. 我在网上找半天解决方案,都不得行.后 ...
- gin+MySQL简单实现数据库查询
利用 gin 项目搭建一个简易的后端系统. 一个简易的 HTTP 响应接口 首先在 go 工作区的终端输入这条指令: go get -u github.com/gin-gonic/gin 将 gin ...
- 【ESP32】制作 Wi-fi 音箱(HTTP + I2S 协议)
用 Wifi 来传输音频数据,会比蓝牙更好.使用蓝牙方式,不管你用什么协议,都会对数据重新编码,说人话就是有损音质,虽然不至于全损.而使用 Wifi 就可以将 PCM 数据直接传输,无需再编码和压缩. ...
- etcd MVCC 存储结构及流程
什么是 MVCC MVCC 是 Multi-Version Concurrency Control 的缩写,即多版本并发控制.它是一种并发控制的方法,用于在数据库系统中实现事务的隔离性.MVCC 是一 ...
- 为WPF框架Prism注册Nlog日志服务
这篇文章介绍了为WPF框架Prism注册Nlog日志服务的方法,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧 无论是Nlog还是Serilog, 它们都提供 ...