国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

vue項(xiàng)目緩存最佳方案

hedzr / 1759人閱讀

需求

在開發(fā)vue的項(xiàng)目中有遇到了這樣一個需求:一個視頻列表頁面,展示視頻名稱和是否收藏,點(diǎn)擊進(jìn)去某一項(xiàng)觀看,可以收藏或者取消收藏,返回的時候需要記住列表頁面的頁碼等狀態(tài),同時這條視頻的收藏狀態(tài)也需要更新, 但是從其他頁面進(jìn)來視頻列表頁面的時候不緩存這個頁面,也就是進(jìn)入的時候是視頻列表頁面的第一頁

一句話總結(jié)一下: pageAList->pageADetail->pageAList, 緩存pageAList, 同時該視頻的收藏狀態(tài)如果發(fā)生變化需要更新, 其他頁面->pageAList, pageAList不緩存

在網(wǎng)上找了很多別人的方法,都不滿足我們的需求

然后我們團(tuán)隊(duì)幾個人搗鼓了幾天,還真的整出了一套方法,實(shí)現(xiàn)了這個需求

實(shí)現(xiàn)后的效果

無圖無真相,用一張gif圖來看一下實(shí)現(xiàn)后的效果吧!!!

操作流程:

首頁->pageAList, 跳轉(zhuǎn)第二頁 ->首頁-> pageAList,頁碼顯示第一頁,說明從其他頁面進(jìn)入pageAList, pageAList頁面沒有被緩存

pageAList, 跳轉(zhuǎn)到第三頁,點(diǎn)擊視頻22 -> 進(jìn)入視頻詳情頁pageADetail,點(diǎn)擊收藏,收藏成功,點(diǎn)擊返回 -> pageAList顯示的是第三頁,并且視頻22的收藏狀態(tài)從未收藏變成已收藏,說明從pageADetail進(jìn)入pageAList,pageAList頁面緩存了,并且更新了狀態(tài)

說明:

二級緩存: 也就是從A->B->A,緩存A

三級緩存:A->B->C->B->A, 緩存A,B

因?yàn)轫?xiàng)目里面絕大部分是二級緩存,這里我們就做二級緩存,但是這不代表我的這個緩存方法不適用三級緩存,三級緩存后面我也會講如何實(shí)現(xiàn)

實(shí)現(xiàn)二級緩存

用vue-cli2的腳手架搭建了一個項(xiàng)目,用這個項(xiàng)目來說明如何實(shí)現(xiàn)
先來看看項(xiàng)目目錄


刪除了無用的components目錄和assets目錄,新增了src/pages目錄和src/store目錄, pages頁面用來存放頁面組件, store不多說,存放vuex相關(guān)的東西,新增了server/app.js目錄,用來啟動后臺服務(wù)

1. 前提條件

項(xiàng)目引入vue,vuex, vue-router,axios等vue全家桶

引入element-ui,只是為了項(xiàng)目美觀,畢竟本人懶癌晚期,不想自己寫樣式

在config/index.js里面配置前端代理

引入express,啟動后臺,后端開3003端口,給前端提供api支持

來看看服務(wù)端代碼server/app.js,非常簡單,就是造了30條數(shù)據(jù),寫了3個接口,幾十行文件直接搭建了一個node服務(wù)器,簡單粗暴解決數(shù)據(jù)模擬問題,會mock用mock也行

const express = require("express")
// const bodyParser = require("body-parser")
const app = express()
let allList = Array.from({length: 30}, (v, i) => ({
  id: i,
  name: "視頻" + i,
  isCollect: false
}))
// 后臺設(shè)置允許跨域訪問
// 前后端都是本地localhost,所以不需要設(shè)置cors跨域,如果是部署在服務(wù)器上,則需要設(shè)置
// app.all("*", function (req, res, next) {
//   res.header("Access-Control-Allow-Origin", "*")
//   res.header("Access-Control-Allow-Headers", "X-Requested-With")
//   res.header("Access-Control-Allow-Methods", "PUT,POST,GET,DELETE,OPTIONS")
//   res.header("X-Powered-By", " 3.2.1")
//   res.header("Content-Type", "application/json;charset=utf-8")
//   next()
// })
app.use(express.json())
app.use(express.urlencoded({extended: false}))
// 1 獲取所有的視頻列表
app.get("/api/getVideoList", function (req, res) {
  let query = req.query
  let currentPage = query.currentPage
  let pageSize = query.pageSize
  let list = allList.slice((currentPage - 1) * pageSize, currentPage * pageSize)
  res.json({
    code: 0,
    data: {
      list,
      total: allList.length
    }
  })
})
// 2 獲取某一條視頻詳情
app.get("/api/getVideoDetail/:id", function (req, res) {
  let id = Number(req.params.id)
  let info = allList.find(v => v.id === id)
  res.json({
    code: 0,
    data: info
  })
})
// 3 收藏或者取消收藏視頻
app.post("/api/collectVideo", function (req, res) {
  let id = Number(req.body.id)
  let isCollect = req.body.isCollect
  allList = allList.map((v, i) => {
    return v.id === id ? {...v, isCollect} : v
  })
  res.json({code: 0})
})
const PORT = 3003
app.listen(PORT, function () {
  console.log("app is listening port" + PORT)
})
2. 路由配置

在路由配置里面把需要緩存的路由的meta添加keepAlive屬性,值為true, 這個想必大家都知道,是緩存路由組件的
在我們項(xiàng)目里面,需要緩存的路由是pageAList,所以這個路由的meta的keepAlive設(shè)置成true,其他路由正常寫,路由文件src/router/index.js如下:

import Vue from "vue"
import Router from "vue-router"
import home from "../pages/home"
import pageAList from "../pages/pageAList"
import pageADetail from "../pages/pageADetail"
import pageB from "../pages/pageB"
import main from "../pages/main"
Vue.use(Router)

export default new Router({
  routes: [
    {
      path: "/",
      name: "main",
      component: main,
      redirect: "/home",
      children: [
        {
          path: "home",
          name: "home",
          component: home
        },
        {
          path: "pageAList",
          name: "pageAList",
          component: pageAList,
          meta: {
            keepAlive: true
          }
        },
        {
          path: "pageB",
          component: pageB
        }
      ]
    },
    {
      path: "/pageADetail",
      name: "pageADetail",
      component: pageADetail
    }
  ]
})
3. vuex配置

vuex的store.js里面存儲一個名為excludeComponents的數(shù)組,這個數(shù)組用來操作需要做緩存的組件

state.js

const state = {
  excludeComponents: [] 
}
export default state

同時在mutations.js里面加入兩個方法, addExcludeComponent是往excludeComponents里面添加元素的,removeExcludeComponent是往excludeComponents數(shù)組里面移除元素

注意: 這兩個方法的第二個參數(shù)是數(shù)組或者組件name

mutations.js

const mutations = {
  addExcludeComponent (state, excludeComponent) {
    let excludeComponents = state.excludeComponents
    if (Array.isArray(excludeComponent)) {
      state.excludeComponents = [...new Set([...excludeComponents, ...excludeComponent])]
    } else {
      state.excludeComponents = [...new Set([...excludeComponents, excludeComponent])]
    }
  },
  // excludeComponent可能是組件name字符串或者數(shù)組
  removeExcludeComponent (state, excludeComponent) {
    let excludeComponents = state.excludeComponents
    if (Array.isArray(excludeComponent)) {
      for (let i = 0; i < excludeComponent.length; i++) {
        let index = excludeComponents.findIndex(v => v === excludeComponent[i])
        if (index > -1) {
          excludeComponents.splice(index, 1)
        }
      }
    } else {
      for (let i = 0, len = excludeComponents.length; i < len; i++) {
        if (excludeComponents[i] === excludeComponent) {
          excludeComponents.splice(i, 1)
          break
        }
      }
    }
    state.excludeComponents = excludeComponents
  }
}
export default mutations
4. keep-alive包裹router-view

將App.vue的router-view用keep-alive組件包裹, main.vue的路由也需要這么包裹,這點(diǎn)非常重要,因?yàn)閜ageAList組件是從它們的router-view中匹配的

這個寫法大家應(yīng)該不會陌生,這也是尤大神官方推薦的緩存方法, exclude屬性值可以是組件名稱字符串(組件選項(xiàng)的name屬性)或者數(shù)組,代表不緩存這些組件,所以vuex里面的addExcludeComponent是代表要緩存組件,addExcludeComponent代表不緩存組件,這里稍微有點(diǎn)繞,請牢記這個規(guī)則,這樣接下來你就不會被繞進(jìn)去了。

App.vue



接下來的兩點(diǎn)設(shè)置非常重要

5. 一級組件

對于需要緩存的一級路由pageAList,添加兩個路由生命周期鉤子beforeRouteEnterbeforeRouteLeave

import {getVideoList} from "../api"
export default {
  name: "pageAList", // 組件名稱,和組件對應(yīng)的路由名稱不需要相同
  data () {
    return {
      currentPage: 1,
      pageSize: 10,
      total: 0,
      allList: [],
      list: []
    }
  },
  methods: {
    getVideoList () {
      let params = {currentPage: this.currentPage, pageSize: this.pageSize}
      getVideoList(params).then(r => {
        if (r.code === 0) {
          this.list = r.data.list
          this.total = r.data.total
        }
      })
    },
    goIntoVideo (item) {
      this.$router.push({name: "pageADetail", query: {id: item.id}})
    },
    handleCurrentPage (val) {
      this.currentPage = val
      this.getVideoList()
    }
  },
  beforeRouteEnter (to, from, next) {
    next(vm => {
      vm.$store.commit("removeExcludeComponent", "pageAList")
      next()
    })
    },
  beforeRouteLeave (to, from, next) {
    let reg = /pageADetail/
    if (reg.test(to.name)) {
      this.$store.commit("removeExcludeComponent", "pageAList")
    } else {
      this.$store.commit("addExcludeComponent", "pageAList")
    }
    next()
  },
  activated () {
    this.getVideoList()
  },
  mounted () {
    this.getVideoList()
  }
}

beforeRouteEnter,進(jìn)入這個組件pageAList之前,在excludeComponents移除當(dāng)前組件,也就是緩存當(dāng)前組件,所以任何路由跳轉(zhuǎn)到這個組件,這個組件其實(shí)都是被緩存的,都會觸發(fā)activated鉤子

beforeRouteLeave: 離開當(dāng)前頁面,如果跳轉(zhuǎn)到pageADetail,那么就需要在excludeComponents移除當(dāng)前組件pageAList,也就是緩存當(dāng)前組件,如果是跳轉(zhuǎn)到其他頁面,就需要把pageAList添加進(jìn)去excludeComponents,也就是不緩存當(dāng)前組件

獲取數(shù)據(jù)的方法getVideoList在mounted或者created鉤子里面調(diào)取,如果二級路由更改數(shù)據(jù),一級路由需要更新,那么就需要在activated鉤子里再獲取一次數(shù)據(jù),我們這個詳情可以收藏,改變列表的狀態(tài),所以這兩個鉤子都使用了

6. 二級組件

對于需要緩存的一級路由的二級路由組件pageADetail,添加beforeRouteLeave路由生命周期鉤子

在這個beforeRouteLeave鉤子里面,需要先清除一級組件的緩存狀態(tài),如果跳轉(zhuǎn)路由匹配到一級組件,再緩存一級組件

beforeRouteLeave (to, from, next) {
    let componentName = ""
    // 離開詳情頁時,將pageAList添加到exludeComponents里,也就是將需要緩存的頁面pageAList置為不緩存狀態(tài)
    let list = ["pageAList"]
    this.$store.commit("addExcludeComponent", list)
    // 緩存組件路由名稱到組件name的映射
    let map = new Map([["pageAList", "pageAList"]])
    componentName = map.get(to.name) || ""
    // 如果離開的時候跳轉(zhuǎn)的路由是pageAList,將pageAList從exludeComponents里面移除,也就是要緩存pageAList
    this.$store.commit("removeExcludeComponent", componentName)
    next()
  }
7.實(shí)現(xiàn)方法總結(jié)

進(jìn)入了pageAList,就在beforeRouteEnter里緩存了它,離開當(dāng)前組件的時候有兩種情況:

1 跳轉(zhuǎn)進(jìn)去pageADetail,在pageAList的beforeRouteLeave鉤子里面緩存pageAList,從pageADetail離開的時候,也有兩種情況

(1) 回到pageAList,那么在pageADetail的beforeRouteLeave鉤子里面緩存了pageAList,所以這就是從pageAList-pageADetail-pageAList的時候,pageAList可以被緩存,還是之前的頁碼狀態(tài)

(2) 進(jìn)入其他路由,在pageADetail的beforeRouteLeave鉤子里面清除了pageAList的緩存

2 跳轉(zhuǎn)到非pageADetail的頁面,在pageAList的beforeRouteLeave鉤子里面清除pageAList的緩存

方案評估

自認(rèn)為用這個方案來實(shí)現(xiàn)緩存,最終的效果非常完美了
缺點(diǎn):

代碼有點(diǎn)多,緩存代碼不好復(fù)用

性能問題:如果在要緩存的一級組件里面寫了activated鉤子,那么從非一級組件對應(yīng)的二級組件進(jìn)入到要緩存的一級組件的時候,會發(fā)送兩次接口請求數(shù)據(jù),mounted里面一次, activated里面一次, 所以如果想追求幾行代碼完美解決緩存問題的,這里就有點(diǎn)無能為力了

項(xiàng)目源碼

項(xiàng)目源碼的github地址,歡迎大家克隆下載

項(xiàng)目啟動與效果演示

npm install安裝項(xiàng)目依賴

npm run server啟動后臺服務(wù)器監(jiān)聽本地3003端口

npm run dev啟動前端項(xiàng)目

三級緩存

上面的方法二級緩存就夠了
上面我們說的是兩個頁面,二級緩存的問題,現(xiàn)在假設(shè)有三個頁面,A1-A2-A3,一步步點(diǎn)進(jìn)去,要求從A3返回到A2的時候,緩存A2,再從A2返回A1的時候,緩存A1,大家可以自己動手研究下,這里就不寫了,其實(shí)就是上面的思路,留給大家研究,大家可以關(guān)注我的微信公眾號,里面有三級緩存的代碼答案。

對不起,還是不能免俗,不管你們?nèi)绾尾粷M,我還是要給我的公眾號打廣告,名字很俗,前端研究中心,但是內(nèi)容不俗,不定期更新優(yōu)質(zhì)前端內(nèi)容:原創(chuàng)或者翻譯國外優(yōu)秀教程,下面是公眾號的二維碼,歡迎大家掃碼加入,一起學(xué)習(xí)和進(jìn)步。

近期優(yōu)質(zhì)文章

和尤雨溪一起進(jìn)階vue

和尤雨溪一起進(jìn)階vue(二)

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/105544.html

相關(guān)文章

  • 前方來報,八月最新資訊--關(guān)于vue2&3的最佳文章推薦

    摘要:哪吒別人的看法都是狗屁,你是誰只有你自己說了才算,這是爹教我的道理。哪吒去他個鳥命我命由我,不由天是魔是仙,我自己決定哪吒白白搭上一條人命,你傻不傻敖丙不傻誰和你做朋友太乙真人人是否能夠改變命運(yùn),我不曉得。我只曉得,不認(rèn)命是哪吒的命。 showImg(https://segmentfault.com/img/bVbwiGL?w=900&h=378); 出處 查看github最新的Vue...

    izhuhaodev 評論0 收藏0
  • vue 在移動端體驗(yàn)上的優(yōu)化解決方案

    摘要:去年年底自己搭了一個在移動端的開發(fā)框架,感覺體驗(yàn)不是很好。路由懶加載首頁終于寫完了,以上這些就是我在移動端體驗(yàn)優(yōu)化的實(shí)戰(zhàn)。去年年底自己搭了一個vue在移動端的開發(fā)框架,感覺體驗(yàn)不是很好。上個星期又要做移動端的項(xiàng)目了。所以我花了兩天時間對之前的那個開發(fā)框架做了以下優(yōu)化 自定義vuex-plugins-loading 路由切換動畫 + keep alive 動態(tài)管理緩存組件 better-sc...

    godlong_X 評論0 收藏0
  • 前端開發(fā)知識點(diǎn)整理

    摘要:前言本文主要是有關(guān)前端方面知識按照目前的認(rèn)知進(jìn)行的收集歸類概括和整理,涵蓋前端理論與前端實(shí)踐兩方面。 前言:本文主要是有關(guān)前端方面知識按照 XX 目前的認(rèn)知進(jìn)行的收集、歸類、概括和整理,涵蓋『前端理論』與『前端實(shí)踐』兩方面。本文會告訴你前端需要了解的知識大致有什么,看上去有很多,但具體你要學(xué)什么,還是要 follow your heart & follow your BOSS。 初衷...

    Blackjun 評論0 收藏0
  • 前端開發(fā)知識點(diǎn)整理

    摘要:前言本文主要是有關(guān)前端方面知識按照目前的認(rèn)知進(jìn)行的收集歸類概括和整理,涵蓋前端理論與前端實(shí)踐兩方面。 前言:本文主要是有關(guān)前端方面知識按照 XX 目前的認(rèn)知進(jìn)行的收集、歸類、概括和整理,涵蓋『前端理論』與『前端實(shí)踐』兩方面。本文會告訴你前端需要了解的知識大致有什么,看上去有很多,但具體你要學(xué)什么,還是要 follow your heart & follow your BOSS。 初衷...

    Sike 評論0 收藏0
  • 前端開發(fā)知識點(diǎn)整理

    摘要:前言本文主要是有關(guān)前端方面知識按照目前的認(rèn)知進(jìn)行的收集歸類概括和整理,涵蓋前端理論與前端實(shí)踐兩方面。 前言:本文主要是有關(guān)前端方面知識按照 XX 目前的認(rèn)知進(jìn)行的收集、歸類、概括和整理,涵蓋『前端理論』與『前端實(shí)踐』兩方面。本文會告訴你前端需要了解的知識大致有什么,看上去有很多,但具體你要學(xué)什么,還是要 follow your heart & follow your BOSS。 初衷...

    tracy 評論0 收藏0

發(fā)表評論

0條評論

hedzr

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<