项目精讲 - keepAlive

<keepAlive> 包裹动态组件时,可以将组件缓存起来。例如我们要浏览商品,可能点击详情后返回仍然在当前位置,不进行更新,这时候就要用到他。

我们首先想到的肯定是可以用他将需要缓存的子组件包裹起来就行了,但是这样在只有几个小页面中的项目可行,项目大了就复杂且难以维护。

我们可以通过路由中配置一个自定义meta属性来决定是否缓存。

  1. 我们可以在根组件中使用v-if的方式来进行判断组件是否缓存。

1.1

js 复制代码
<keep-alive>
  <router-view v-if="$route.meta.keepAlive"></router-view>
</keep-alive>
<router-view v-if="!$route.meta.keepAlive"></router-view>

1.2

js 复制代码
<router-view v-slot="{ Component, route }"> // 这里解构出路由对象和组件对象,不是插槽,是新用法
  <keep-alive>
    <component :is="Component" v-if="route.meta.keepAlive"/>
  </keep-alive>
  <component :is="Component" v-if="!route.meta.keepAlive"/>
</router-view>
  1. 通过 keepAlive自带的属性进行绑定。 他有三个属性可以进行设置

我们通过include属性来进行一个动态绑定。当我们是从进入详情页面返回的时候,我们不进行缓存,当我们从最初页面平级跳转,我们清除缓存。

我们现在组件使用给keepAlive包裹一下:

js 复制代码
<template>
  <router-view v-slot="{ Component }">
    <keep-alive :include="cachedComponents">
      <component :is="Component" />
    </keep-alive>
  </router-view>
</template>

<script setup>
import { ref, watch } from 'vue'
import { useRoute } from 'vue-router'

const cachedComponents = ref([])
const route = useRoute()

watch(route, (newVal,old) => {
  if (newVal.meta.keepAlive && !cachedComponents.value.includes(newVal.name)) {
     if(newVal.meta.deep > )
    cachedComponents.value.push(newVal.name)
  }
})
</script>

我们先来设置一下路由:

js 复制代码
// router/index.ts
import { createRouter, createWebHistory, type RouteRecordRaw } from 'vue-router'

const routes: RouteRecordRaw[] = [
  {
    path: '/list',
    name: 'List',
    component: () => import('@/views/List.vue'),
    meta: {
      depth: 1 // 添加层级标记
    }
  },
  {
    path: '/detail/:id',
    name: 'Detail',
    component: () => import('@/views/Detail.vue'),
    meta: {
      depth: 2
    }
  },
  {
    path: '/settings',
    name: 'Settings',
    component: () => import('@/views/Settings.vue'),
    meta: {
      depth: 1
    }
  }
]

const router = createRouter({
  history: createWebHistory(),
  routes
})

// 缓存管理逻辑
const cachePages = ref<string[]>([])

router.beforeEach((to, from) => {
  // 判断是否是前进更深层级
  const isEnteringDeeper = (to.meta.depth as number) > (from.meta.depth as number)
  
  // 判断是否是返回操作
  const isGoingBack = (to.meta.depth as number) < (from.meta.depth as number)

  if (isEnteringDeeper) {
    // 进入更深层级时缓存前一个页面
    if (!cachePages.value.includes(from.name as string)) {
      cachePages.value.push(from.name as string)
    }
  } else if (!isGoingBack) {
    // 平级切换时移除前一个页面的缓存
    const index = cachePages.value.indexOf(from.name as string)
    if (index > -1) {
      cachePages.value.splice(index, 1)
    }
  }
})

keepAlive有两个生命周期钩子,onactived 和 onDeactived。 当我们的被缓存的页面显示时,会触发onactived钩子,隐藏时会触发 onDeactived钩子,触发顺序为 onCreated onMounted on Deactived。

路由守卫会在onDeactived之前执行。确保一些逻辑的完成。

我们可以利用onactived钩子来进行一些数据的更新。

相关推荐
萧曵 丶1 小时前
Vue3组件通信全方案
前端·javascript·vue.js·typescript·vue3
前端那点事1 小时前
双Token无感刷新:Vue3 + Axios 企业级完整实现
前端·vue.js
前端那点事1 小时前
Vue Token鉴权避坑指南|5步完整实现(从生成到失效全解析)
前端·vue.js
前端那点事1 小时前
企业级Vue前端鉴权方案全解析|从Token到OAuth2.0,覆盖多端适配+权限管控
前端·vue.js
亲亲小宝宝鸭1 小时前
从Vben-Admin里面学习hooks
前端·vue.js
天蓝色的鱼鱼2 小时前
当AI开始替我写代码,我还要纠结选Vue还是React吗?
vue.js·react.js·ai编程
QQ_1880838003 小时前
python+flask+vue在线宠物医疗预约平台的设计与实现_b5z03zls
vue.js·python·flask
用户66885998476617 小时前
Vue 3.0安装与使用
vue.js
前端之虎陈随易20 小时前
2年没用Nodejs了,Bun很香
linux·前端·javascript·vue.js·typescript
前端 贾公子1 天前
解决浏览器端 globalThis is not defined 报错
前端·javascript·vue.js