在 Vue 里如何优雅的清除一个定时器?

「本文已参与好文召集令活动,点击查看后端、大前端双赛道投稿,2万元奖池等你挑战!javascript

前言

清除定时器,相信有至关一部分人是这么写的:前端

export default {
  data() {
    return {
      timer: null
    }
  },
  
  mounted() {
    this.timer = setInterval(() => {
      console.log('setInterval')
    }, 2000)
  },
  
  beforeDestroy() {
    clearInterval(this.timer)
  }
}
复制代码

这是一段常见的代码,至少我身边的好几个小伙伴(1-3年经验的都有)都是这么写的,这里存在3个不优雅的问题:java

  • clearInterval 后没有清空 timer 为 null。
  • 开启定时器和清除定时器的代码分散开在两个地方,有损可读性/维护性,用尤大大的话说,这使得咱们比较难于程序化地清理咱们创建的东西。
  • timer 被定义在 data 里,实际上 timer 不须要什么响应式操做,定义在 data 里是不必的,反而形成性能浪费。

优化

直接上代码:后端

export default {
  data() {
    return {
    }
  },
  
  mounted() {
    let timer = setInterval(() => {
      console.log('setInterval')
    }, 2000)
    this.$once('hook:beforeDestroy', () => {
      clearInterval(timer)
      timer = null
    })
  }
}
复制代码

这里使用 hook 监听 beforeDestroy 生命周期,这样 timer 只需被定义在生命周期里,以上的问题就所有解决了。缓存

衍生问题:beforeDestroy 没有触发?

在后台系统中,咱们经常会设置页面缓存,而当路由被 keep-alive 缓存时是不走 beforeDestroy 生命周期的,因此有些小伙伴觉得在 beforeDestroy 清除定时器就完事了,甚至都没有检查一下,实际上定时器并无被清除掉。知道了缘由也就好解决了,借助 activated 和 deactivated 这两个生钩子:markdown

export default {
  data() {
    return {
    }
  },
  
  mounted() {
    let timer = setInterval(() => {
      console.log('setInterval')
    }, 2000)
    this.$on('hook:activated', () => {
      if (timer === null) { // 避免重复开启定时器
        timer = setInterval(() => {
          console.log('setInterval')
        }, 2000)
      }
    })
    this.$on('hook:deactivated', () => {
      clearInterval(timer)
      timer = null
    })
  }
}
复制代码

这里须要注意一下,因为缓存缘由,因此须要用 $on 而不是 $once,否则执行一次后就不会再触发了。post

感谢

感谢你的阅读,若是本文对你有什么帮助,别忘了点赞支持一下❤️。性能

本文若是有什么错误或不足,欢迎评论区指正。优化

相关文章
相关标签/搜索