vue多级路由缓存(keep-alive)失效的解决方案

10,404 阅读5分钟

最近在开发一个基于Vue3的后台管理系统,整个项目结构参考了 vue-element-plus-admin(简称 admin),在处理页面缓存方面,它使用了 keep-alive 结合 router-view 实现,这也是大多后台系统的方向。由于 admin 的router 是一个扁平结构,故不存在缓存失效问题。而我的系统的router是个树结构,涉及多层 router-view 嵌套,这样 admin 中简单的实现就满足不了我的需求了。

如何解决多级路由缓存失效问题?是本文讨论的重点,如果你也有同我一样的烦恼,不妨读读本文😆。

小编简化系统由这几个部分组成:

  • 基础布局:Layout.vue
  • 路由配置:router.js
  • 路由页面:Home.vue、About.vue、Theme1.vue、Theme2-1.vue、Theme3、NestRouterView.vue

方案一(树结构转扁平结构)

项目中的 routes 管理如下,很经典的树结构。根节点是一个基础布局,所有页面都基于这个布局展示。二级路由上,借助了router-view组件去承载子路由。我们给像 NestRouterView.vue 此类功能组件统称为『中间件』。

const views = {
  path: "/",
  component: () => import("@/views/Layout.vue"),
  children: [
    {
      name: "Home",
      path: "",
      component: () => import("@/views/Home.vue"),
      meta: {
        title: "首页",
        keepAlive: false
      }
    },
    {
      name: "About",
      path: "about",
      component: () => import("@/views/About.vue"),
      meta: {
        title: "关于",
        keepAlive: true
      }
    },
    {
      path: "",
      component: () => import("@/views/NestRouterView.vue"),
      children: [
        {
          name: "Theme1",
          path: "theme/1",
          component: () => import("@/views/Theme1.vue"),
          meta: {
            title: "主题1",
            keepAlive: true
          }
        },
        {
          name: "Theme3",
          path: "theme/3",
          component: () => import("@/views/Theme3.vue"),
          meta: {
            title: "主题3",
            keepAlive: false
          }
        },
        {
          path: "",
          component: () => import("@/views/NestRouterView.vue"),
          children: [
            {
              name: "Theme2-1",
              path: "theme/2/1",
              component: () => import("@/views/Theme2-1.vue"),
              meta: {
                title: "主题2-1",
                keepAlive: true
              }
            }
          ]
        }
      ]
    }
  ]
}

<template>
  <router-view />
</template>
<script>
export default {
  name: 'NestRouterView'
}
</script>

实际上是否是多级路由对于项目或者业务上来讲一点都不影响,只是界面显示需要,让用户能更快的找到自己需要的功能页面而已。这样的话,显示的路由保留多级的,实际上 router 维护成一级,将显示菜单和业务 router 分离开。放弃原有树结构,保持 router 扁平化,我们就得到了以下的 routes,不存在复杂嵌套,也不构建多余的『中间件』组件。

const views = {
  path: "/",
  component: () => import("@/views/Layout.vue"),
  children: [
    {
      name: "Home",
      path: "",
      component: () => import("@/views/Home.vue"),
      meta: {
        title: "首页",
        keepAlive: false
      }
    },
    ......
    {
      name: "Theme1",
      path: "theme/1",
      component: () => import("@/views/Theme1.vue"),
      meta: {
        title: "主题1",
        keepAlive: true
      }
    },
    ......
    {
      name: "Theme2-1",
      path: "theme/2/1",
      component: () => import("@/views/Theme2-1.vue"),
      meta: {
        title: "主题2-1",
        keepAlive: true
      }
    }
  ]
}

有人说我这项目里有几十、上百条路由配置,让我手动打平,我这一天都别干活了。好办!既然『中间件』只是承载子路由功能,那我们就可以在扁平化过程中不考虑它。

const formatRouter = (routes, newRoutes = []) => {
  routes.forEach((item) => {
    if (item.children && item.children.length > 0) {
      formatRouter(item.children, newRoutes);
    } else {
      newRoutes.push(item);
    }
  });
  return newRoutes;
};

views.children = formatRouter(views.children);

最终效果如下,除了首页主题3页面都进行了缓存。

Video_2022-05-18_160009.gif

完整代码

方案二

方案一的实现直接明了,却存在很多局限性。特别是『中间件』文件(NestRouterView.vue),要是承载着业务功能,那就不能这么简单粗暴解决了。接下来我们来看看方案二的解法。

方案二的实现得力于3个函数:

  • findRouterWith
  • getFileNameByFunContext
  • getCachesByRoutes

findRouterWith 以『中间件』组件名作为查询条件来获取所有以其作为载体的页面。

// 找到以组件名 name 作为“中间件”的所有路由
export const findRouterWith = (name) => {
  const fn = (routes, newRoutes = []) => {
    routes.forEach((item) => {
      if (item.children && item.children.length > 0) {
        if (getFileNameByFunContext(item.component.toString()) === name) {
          newRoutes.push(item.children);
        }
        fn(item.children, newRoutes);
      }
    });
    return newRoutes.flat(Infinity);
  };
  return fn;
};

由于在 router.js 中配置的 component 是一个箭头函数,我们无法直接获取『中间件』组件名,故采用通过函数内容来获取组件名。

{
  path: "/",
  component: () => import("@/views/NestRouterView.vue")
}

component.toString()

function component() {
      return __webpack_require__.e(/*! import() */ 4).then(__webpack_require__.bind(null, /*! @/views/NestRouterView.vue */ "./src/views/NestRouterView.vue"));
    }

通过观察,我们发现好提取的组件名位于双引号之间,这样我们就可以先通过提取组件路径,然后再匹配文件名。如果你有更好的写法也可以在评论区告诉我。

// 通过函数内容获取文件名
export const getFileNameByFunContext = (str) => {
  const [file = ""] = str.match(/".+"/);
  return file.replace(/(.*\/)*([^.]+).*/gi, "$2");
};

getCachesByRoutes 函数以路由数组入参,返回需进行缓存的一个数组。

函数先是定义了一个 children 变量来存储『中间件』,接着过滤掉路由中所有含有子集(借助了『中间件』)或是不需要缓存的项,将含有子集项的值存储于 children 中,最后的返回值是由已过滤路由和『中间件』集组成。

// 获取需进行缓存的页面
export const getCachesByRoutes = (routes = []) => {
  const children = [];
  const caches = routes
    .filter((o) => {
      // 有children说明进行了路由嵌套,需记录“中间件”
      if (o.children) {
        children.push(o.component);
      }
      // 过滤掉“中间件”和不需要缓存的组件
      return !o.children && o.meta.keepAlive;
    })
    .map((o) => o.name);

  if (children.length > 0) {
    // 路由嵌套的组件也需include
    children.forEach((fun) => {
      caches.push(getFileNameByFunContext(fun.toString()));
    });
  }
  return [...new Set(caches)]; // 排重
};

以『中间件』(NestRouterView.vue)为例,此时的 NestRouterView 不仅仅是容器,还承载功能。首先,我们看 router.js 文件,routes 比原来多了层嵌套,情况也就更复杂;接着 findRouterWith 获取以 NestRouterView 为父组件的所有路由,最后通过 getCachesByRoutes 来获取需在本组件进行缓存的数组。

router.js

const routes = [
  {
    path: "/",
    component: () => import("@/views/Layout.vue"),
    children: [
      ...... // 同上
      {
        path: "",
        component: () => import("@/views/NestRouterView.vue"),
        children: [
          ......// 同上
          {
            path: "",
            component: () => import("@/views/NestRouterView.vue"),
            children: [
              {
                name: "Theme2-1",
                path: "theme/2/1",
                component: () => import("@/views/Theme2-1.vue"),
                meta: {
                  title: "主题2-1",
                  keepAlive: true
                }
              }
            ]
          }
        ]
      }
    ]
  }
];

NestRouterView.vue

<template>
  <div>
    <fieldset>
      <button @click="goBack">返回</button>
    </fieldset>
    <router-view v-slot="{ Component }">
      <keep-alive :include="caches">
        <component :is="Component" />
      </keep-alive>
    </router-view>
  </div>
</template>
<script>
import { routes } from "@/router";
import { findRouterWith, getCachesByRoutes } from "@/utils";

export default {
  name: "NestRouterView",
  setup() {
    // 找到所有以 NestRouterView 组件作为路由嵌套的页面
    const target = findRouterWith("NestRouterView")(routes);
    const caches = getCachesByRoutes(target);
    const goBack = () => window.history.back();

    return {
      caches,
      goBack,
    };
  },
};
</script>

image.png

最终的caches

最终效果如下,除了首页主题3页面都进行了缓存。但有个地方比较怪异,就是主题2-1页面出现了两个 【返回】功能。那是因为 NestRouterView 嵌套所致,你可以通过设置 prop 的方式来达到页面统一。

ok.gif

{
    name: "Theme2-1",
    path: "theme/2/1",
    component: () => import("@/views/Theme2-1.vue"),
    props: () => ({ showBackBtn: false }),
    meta: {
      title: "主题2-1",
      keepAlive: true
    }
}
export default {
  name: "NestRouterView",
  props: {
    showBackBtn: {
      type: Boolean,
      default: true,
    },
  },
}

image.png

完整代码

小结

方案一是目前互联网中最常见的解法,个人觉得还是存在较大的局限性。方案二是小编在项目开发过程中摸索出的解法,可扩展行强。强推!!!若你还有更好的方案,可分享于我。

若你在使用 keep-alive 中有疑惑,可移步姊妹篇《看keep-alive如何在项目中失效

看过姊妹篇的朋友可能记得小编说过,vue-router 的 name 属性是用于路由跳转,不是用于 keep-alive,但在本文的方法却取了 vue-router 的 name 用于 include 中。这是因为小编统一了组件和 vue-router 的 name,更有利于项目的维护。

本文首发掘金平台,来源个人博客