一劳永逸搞定列表页面缓存

背景介绍

在前端日常项目开发中,大家应该都有遇到过这样的需求:当我们在列表页输入条件进行数据筛选时,我们可能需要根据筛选后的结果进入到某一条的详情页,当我们看完详情页内容,再返回列表页,此时我们希望还是之前的筛选结果,包括条件和分页等。

在实际开发中,改变页面导航的路由都会重新匹配页面,初始化页面的数据状态,无法直接达到这种效果。但Vue给我们提供了“秘密武器”-keep-alive,可以实现页面的数据缓存。

其实以前也写过类似的实现页面缓存的方案:Vue-Router 实现前端页面缓存,是通过keep-aliveinclude属性控制要缓存的页面。但是这种方式不够灵活,一旦缓存了页面,从其他页面(非详情页面)跳入到列表页时数据也不会重新更新,需要在组件路由守卫beforeRouteEnter中去做判断处理,十分麻烦。

因此,经过参考探索,今天来分享一个一劳永逸的实现页面缓存的方案。

实现思路

总体思路:

只要是列表页(需要使用缓存的页面),进入之前,都将其添加到要缓存的页面中,离开列表页时判断新(to)路由是否是该列表页指定的详情页(配置路由时通过 cacheTo 字段来指定),如果不是就清空缓存。

思路分析

根据总体思路,主要分为以下四种情况:

  1. fromto都不是列表页
    • 都不需要缓存,清空以前的缓存
  2. fromto都是列表页
    • 2.1-如果to不在from的配置中,清空缓存,新增to缓存
    • 2.2-如果tofrom的配置中,保留from缓存,新增to缓存
  3. to是列表页,from不是
    • 3.1-from不在to的配置中,清空缓存,新增to缓存
    • 3.2-fromto的配置中,不做处理(因为 3.1 已经缓存了 to)
  4. from是列表页,to不是
    • 4.1-to不在from的配置中,清空缓存
    • 4.2-tofrom的配置中,不做处理(因为 from 已经在 3.1 时缓存)

以上思路就是日常所有的场景,理解起来可能会比较绕,结合实际场景理解就会发现所有情况都已经覆盖到。

具体实现

前置须知

  • 在路由表配置中维护一个cacheTo字段,如果配置了该字段就表示该路由为列表页。
  • 控制缓存的逻辑,我们单独在一个文件内完成,作为组件单独管理。为了避免include的管理更加简单,就不适用vuex,这里使用Vue.observable
  • 为了避免维护keep-aliveinclude属性对代码造成侵入,这里使用函数式组件。
  • 为了在管理缓存的组件内能获取到tofrom,需要在组件内注册一个beforeEach钩子,vue-routerbeforeEach钩子是可以重复注册的,按照注册顺序执行。

实现细节

// keep-alive-rorute.js
import Vue from "vue";
/**
 * 创建存储缓存仓库
 */
// 存储需要缓存页面路由
const cacheState = Vue.observable({
  caches: [],
});
// 清理路由缓存
const clearCache = () => {
  if (!cacheState.caches.length) return;
  cacheState.caches = [];
};
// 新增路由缓存
const addCache = (name) => cacheState.caches.push(name);

const defaultHook = (to, from, next) => next();
// 路由进入钩子函数
export const beforeRouterEachHook = (hook = defaultHook) => {
  return (to, from, next) => {
    /**
     * 思路:只要是类列表页,进入之前都将其缓存,离开时判断新路由是否是该类列表页指定的类详情页,如果不是就清除缓存。
     */
    // to页面路由
    const toRouteName = to.name;
    // to页面为列表页时配置的to路由集合
    const toCacheRoutes = (to.meta || {}).cacheTo;
    // to页面是否是列表页
    const isToPageList = toCacheRoutes && toCacheRoutes.length;
    // from页面路由名称
    const fromRouteName = from.name;
    // form页面为列表页时,配置的to页面路由集合
    const fromCacheRoutes = (from.meta || {}).cacheTo;
    // from页面是否是列表页
    const isFromPageList = fromCacheRoutes && fromCacheRoutes.length;

    // 1如果to,from都不是列表页,清除所有缓存
    if (!isToPageList && !isFromPageList) {
      clearCache();
    } else if (isToPageList && isFromPageList) {
      // 2如果to,from都是列表页
      // 如果to列表页在from列表页的配置中就要缓存,否则不用
      if (fromCacheRoutes.indexOf(toRouteName) < 0) {
        clearCache();
      }
      if (to.matched && to.matched.length > 2) {
        // 三级路由时,缓存父子两级路由
        to.matched.map((element) => {
          if (element.name) {
            addCache(element.name);
          }
        });
      } else {
        // 二级路由缓存当前路由
        addCache(toRouteName);
      }
    } else if (isToPageList) {
      // 3如果to是列表页
      // 如果from不在to的配置中,清除所有缓存,同时缓存新路由
      if (toCacheRoutes.indexOf(fromRouteName) < 0) {
        clearCache();
        if (to.matched && to.matched.length > 2) {
          // 三级路由时缓存父子两级路由
          to.matched.map((element) => {
            if (element.name) {
              addCache(element.name);
            }
          });
        } else {
          // 二级路由缓存当前路由
          addCache(toRouteName);
        }
      }
    } else if (isFromPageList) {
      // 4如果from页面是列表页
      // to不在from的配置中清空缓存
      if (fromCacheRoutes.indexOf(toRouteName) < 0) {
        clearCache();
      }
    }
    return hook(to, from, next);
  };
};

// 缓存路由组件
export const KeepAliveRoute = {
  install(Vue) {
    const component = {
      name: "KeepAliveRoute",
      functional: true,
      render(h, params) {
        return h(
          "keep-alive",
          { props: { include: cacheState.caches } },
          params.children
        );
      },
    };
    Vue.component("KeepAliveRoute", component);
  },
};

项目使用

  • main.js中注册组件并执行缓存逻辑
// main.js
import Vue from "vue";
import App from "./App.vue";
import router from "./router";
// 引入
import { beforeRouterEachHook, KeepAliveRoute } from "./keep-alive-rorute";
// 注册
Vue.use(KeepAliveRoute);
// 路由守卫中执行缓存逻辑
router.beforeEach(beforeRouterEachHook());
new Vue({
  router,
  store,
  render: (h) => h(App),
}).$mount("#app");
  • 在需要缓存的路由展示区使用缓存组件
// Home.vue
<template>
  <transition name="fade-transform" mode="out-in">
    // 使用缓存组件
    <KeepAliveRoute>
      <router-view></router-view>
    </KeepAliveRoute>
  </transition>
</template>
  • 在路由配置表中配置列表页
[
  {
    path: "/list",
    name: "List",
    meta: {
     // 配置需要进入列表页使用缓存数据的页面
      cacheTo: ["Detail"],
    },
    ....
  },
  {
      path: 'detail',
      name: 'Detail',
      ......
  },
];

注意

  • 路由配置必须使用name字段,且和对应的组件name对应。
  • 支持多层级嵌套的父子路由缓存。只需要在父子组件的router-view外面包上KeepAliveRoute组件即可。

查看更多内容,请关注微信公众号:柒荤捌素

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容

  • 今天感恩节哎,感谢一直在我身边的亲朋好友。感恩相遇!感恩不离不弃。 中午开了第一次的党会,身份的转变要...
    迷月闪星情阅读 10,562评论 0 11
  • 彩排完,天已黑
    刘凯书法阅读 4,205评论 1 3
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 124,795评论 2 7