vue keepalive 前进刷新后退不刷新终极解决方案

2020-01-17 08:44:57

关于vue keeplive的刷新缓存终极解决方案,就看下面了,全栈网其他关于解决keeplive的大家就当作参考吧。

另外,我们做路由的时候要有意的根据页面等级做出路由的长度 比如 /a是一级的页面/a/b是二级的页面,下面的文章大家也可以通过判断path的长度来计算rank值,不用有意自定了

这样做的好处有两点,一个就是前进刷新,后退不刷新,还有就是,如果我们做页面进出效果的时候也能排上用场。

 遇到这一个个问题  需要是这样的 Vue里面的不刷新问题

页面分为: A 主页  B列表页  C 详情页

A  beforeRouteLeave 时设置 to.meta.keepAlive = false (A 进入 B页面时), 不缓存 B列表页

B beforeRouteLeave 时 (B -> C)列表页进详情页时 from.meta.keepAlive =true 缓存B列表页

C详情页 返回 B列表页 ,此时B列表页缓存不生效,也就是第一次不生效

总结问题原因:

当进入B列表页后,keepAlive 为 true 时,缓存生效

当离开B列表页时 设置 keepAlive 为 true  时 ,缓存不生效

这就是很头疼的问题,查阅相关资料找到

解决

在app.vue 页面

<transition :name="transitionName">
      <keep-alive>
        <!-- v-if="isRouterALive" :include="[ keepData ]" -->
        <router-view class="child-view" v-if="$route.meta.keepAlive" ></router-view>
      </keep-alive>
    </transition>
    <transition :name="transitionName">
      <router-view class="child-view" v-if="!$route.meta.keepAlive"></router-view>
    </transition>

路由页面配置:

{
    path: '/activities/create_proto',
    name: 'create_proto',
    meta: {
      index: 2,
      title: '活动创建',
      keepAlive: false, //此组件不需要被缓存
      rank:1,
    },
    component: () =>
      import (/* webpackChunkName: "create_proto" */'@/pages/activities/create_proto')
  },
  {
    path: '/activities/create/:id',
    name: 'create',
    meta: {
      index: 2,
      title: '活动创建',
      keepAlive: true, //此组件不需要被缓存
      isBack:false, //用于判断上一个页面是哪个
      rank:1.5,
    },
    component: () =>
      import (/* webpackChunkName: "create" */'@/pages/activities/create')
  },

  {
    path: '/activities/createMask',
    name: 'createMask',
    meta: {
      index: 2,
      title: '组',
      keepAlive: false, //此组件不需要被缓存
      isBack:false, //用于判断上一个页面是哪个
      rank:1.5,
    },
    component: () =>
      import (/* webpackChunkName: "createMask" */'@/pages/activities/createMask')
  },

让B页面始终是缓存,

然后在B页面 就是 create页面 里:

beforeRouteEnter(to,from,next){
    //来自editNotic的不缓存
      if(from.name === 'createMask' && to.name === 'create'){
        to.meta.isBack = true;
      }
      next();
    },
    activated() {
    //   console.log(this.$route.meta)
      if(!this.$route.meta.isBack || this.isFirstEnter){
        // 如果isBack是false,表明需要获取新数据,否则就不再请求,直接使用缓存的数据
        this.getApi();
      }
      // 恢复成默认的false,避免isBack一直是true,导致下次无法获取数据
      this.$route.meta.isBack=false;
      //this.isBack = false;
      this.isFirstEnter = false;
    },
    beforeCreate() {
      this.$loading.open();
    },

    mounted(){
      this.isFirstEnter = true;
      this.getApi();
    },

解释一波

缓存的页面 created 会执行只有一次,activated每次都会执行 ,

created 里面做 第一次 isFirstEnter = true(由于页面被缓存,所以一直生效),之后再activated 里面做判断

只有 “不是返回回来的” 和 “第一次进来的” 就刷新数据, 并且要在下面 都设为false, 以免缓存各标识不对,

在进入 “列表页” 时,通过router钩子函数 beforeRouteEnter做判断,

详情页过来的设 isBack 为true,即不刷新页面

 

但第一次都是不生效的,查阅了github 上的大佬方法,就是强制清除B缓存,当B页面离开去到A页面,用rank 来比较

在main.js 中 写入:

Vue.mixin({
  beforeRouteLeave: function (to, from, next) {
    console.log(to)
    if (from && from.meta.rank && to.meta.rank && from.meta.rank > to.meta.rank) { //如果返回上一层,则摧毁本层缓存。
      if (this.$vnode && this.$vnode.data.keepAlive) {
        if (this.$vnode.parent && this.$vnode.parent.componentInstance && this.$vnode.parent.componentInstance.cache) {
          if (this.$vnode.componentOptions) {
            var key = this.$vnode.key == null ?
              this.$vnode.componentOptions.Ctor.cid + (this.$vnode.componentOptions.tag ? `::${this.$vnode.componentOptions.tag}` : '') :
              this.$vnode.key;
            var cache = this.$vnode.parent.componentInstance.cache;
            var keys = this.$vnode.parent.componentInstance.keys;
            if (cache[key]) {
              if (keys.length) {
                var index = keys.indexOf(key);
                if (index > -1) {
                  keys.splice(index, 1);
                }
              }
              delete cache[key];
            }
          }
        }
      }
      this.$destroy();
    }
    next();
  },
})

就是强制清除缓存,哇,这个问题搞了半天,

然后就解决了前进刷新,后退不刷的问题,

当然还可以结合vuex 来做到效果,但数据比较大的话还是适合 我写的这种吧,哈哈哈


  • 2021-04-15 10:10:00

    Puppeteer 系列踩坑日志—3—开启支持插件

    在使用puppeteer自动化的过程中,会发现其实开启的chrome往往自动禁用了插件功能,如果我们想在自动化测试的过程中,再去使用一些常用的插件提升效率(偷懒)的话,就行不通了,其实解决办法还是有的,我们今天就来讲解这个问题。

  • 2021-04-15 10:11:17

    Puppeteer拦截修改返回值

    page.setRequestInterception(true)拦截器的使用方法和场景 现附上Puppeteer的Api的链接https://github.com/GoogleChrome/puppeteer/blob/master/docs/api.md

  • 2021-04-15 10:32:18

    怎么给 headless chrome添加cookies

    In puppeter you have access to the session cookies through page.cookies(). So once you log in, you could get every cookie and save it in a json file:

  • 2021-04-15 10:51:21

    如何通过Devtools协议拦截和修改Chrome响应数据

    在日常研究中,我们经常碰到大量JavaScript代码,我们首先要深入分析才能了解这些代码的功能及具体逻辑。这些代码代码可能会被恶意注入到页面中,可能是客户送过来需要我们帮忙分析的脚本,也可能是我们的安全团队在网页上找到的引用了我们服务的某些资源。这些脚本通常代码量不大、经过混淆处理,并且我们总是需要经过多层修改才能继续深入分析。

  • 2021-04-19 10:54:39

    block和delegate的区别

    代理 可读性高 大部分可以属性 block 写的代码少 一般作为参数 通知 占用资源

  • 2021-04-19 11:00:23

    浅谈block和delegate的使用

    委托是协议的一种,顾名思义,就是委托他人帮自己去做事。委托是给一个对象提供机会对另一个对象中的变化做出反应或者影响另一个对象的行为。其基本思想是:两个对象协同解决问题,并且打算在广泛的情形中重用。委托指向另一个对象(即它的委托)的引用,并在关键时刻给委托发消息。消息可能只是通知委托发生了某件事情,给委托提供机会执行额外的处理,或者消息可能要求委托提供一些关键的信息以控制所发生的事情。委托的作用主要有两个,一个是传值,一个是传事件。

  • 2021-04-19 11:36:44

    iOS 组件实现方案

    什么才是好架构,为什么要组件,组件设计的优点