国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

Vue3 除了 keep-alive,還有哪些實(shí)現(xiàn)頁(yè)面緩存的方法

這篇具有很好參考價(jià)值的文章主要介紹了Vue3 除了 keep-alive,還有哪些實(shí)現(xiàn)頁(yè)面緩存的方法。希望對(duì)大家有所幫助。如果存在錯(cuò)誤或未考慮完全的地方,請(qǐng)大家不吝賜教,您也可以點(diǎn)擊"舉報(bào)違法"按鈕提交疑問。

有這么一個(gè)需求:列表頁(yè)進(jìn)入詳情頁(yè)后,切換回列表頁(yè),需要對(duì)列表頁(yè)進(jìn)行緩存,如果從首頁(yè)進(jìn)入列表頁(yè),就要重新加載列表頁(yè)。

對(duì)于這個(gè)需求,我的第一個(gè)想法就是使用keep-alive來緩存列表頁(yè),列表和詳情頁(yè)切換時(shí),列表頁(yè)會(huì)被緩存;從首頁(yè)進(jìn)入列表頁(yè)時(shí),就重置列表頁(yè)數(shù)據(jù)并重新獲取新數(shù)據(jù)來達(dá)到列表頁(yè)重新加載的效果。

但是,這個(gè)方案有個(gè)很不好的地方就是:如果列表頁(yè)足夠復(fù)雜,有下拉刷新、下拉加載、有彈窗、有輪播等,在清除緩存時(shí),就需要重置很多數(shù)據(jù)和狀態(tài),而且還可能要手動(dòng)去銷毀和重新加載某些組件,這樣做既增加了復(fù)雜度,也容易出bug。

接下來說說我的想到的新實(shí)現(xiàn)方案(代碼基于Vue3)。

keep-alive 緩存和清除

keep-alive 緩存原理:進(jìn)入頁(yè)面時(shí),頁(yè)面組件渲染完成,keep-alive 會(huì)緩存頁(yè)面組件的實(shí)例;離開頁(yè)面后,組件實(shí)例由于已經(jīng)緩存就不會(huì)進(jìn)行銷毀;當(dāng)再次進(jìn)入頁(yè)面時(shí),就會(huì)將緩存的組件實(shí)例拿出來渲染,因?yàn)榻M件實(shí)例保存著原來頁(yè)面的數(shù)據(jù)和Dom的狀態(tài),那么直接渲染組件實(shí)例就能得到原來的頁(yè)面。

keep-alive 最大的難題就是緩存的清理,如果能有簡(jiǎn)單的緩存清理方法,那么keep-alive 組件用起來就很爽。

但是,keep-alive 組件沒有提供清除緩存的API,那有沒有其他清除緩存的辦法呢?答案是有的。我們先看看 keep-alive 組件的props:

include?-?string?|?RegExp?|?Array。只有名稱匹配的組件會(huì)被緩存。
exclude?-?string?|?RegExp?|?Array。任何名稱匹配的組件都不會(huì)被緩存。
max?-?number?|?string。最多可以緩存多少組件實(shí)例。

從include描述來看,我發(fā)現(xiàn)include是可以用來清除緩存,做法是:將組件名稱添加到include里,組件會(huì)被緩存;移除組件名稱,組件緩存會(huì)被清除。根據(jù)這個(gè)原理,用hook簡(jiǎn)單封裝一下代碼:

import?{?ref,?nextTick?}?from?'vue'

const?caches?=?ref<string[]>([])

export?default?function?useRouteCache?()?{
??//?添加緩存的路由組件
??function?addCache?(componentName:?string?|?string?[])?{
????if?(Array.isArray(componentName))?{
??????componentName.forEach(addCache)
??????return
????}
????
????if?(!componentName?||?caches.value.includes(componentName))?return

????caches.value.push(componentName)
??}

??//?移除緩存的路由組件
??function?removeCache?(componentName:?string)?{
????const?index?=?caches.value.indexOf(componentName)
????if?(index?>?-1)?{
??????return?caches.value.splice(index,?1)
????}
??}
??
??//?移除緩存的路由組件的實(shí)例
??async?function?removeCacheEntry?(componentName:?string)?{????
????if?(removeCache(componentName))?{
??????await?nextTick()
??????addCache(componentName)
????}
??}
??
??return?{
????caches,
????addCache,
????removeCache,
????removeCacheEntry
??}
}

hook的用法如下:

<router-view v-slot="{ Component }">
  <keep-alive :include="caches">
    <component :is="Component" />
  </keep-alive>
</router-view>

<script setup lang="ts">
import useRouteCache from './hooks/useRouteCache'
const { caches, addCache } = useRouteCache()

<!-- 將列表頁(yè)組件名稱添加到需要緩存名單中 -->
addCache(['List'])
</script>

清除列表頁(yè)緩存如下:

import?useRouteCache?from?'@/hooks/useRouteCache'

const?{?removeCacheEntry?}?=?useRouteCache()
removeCacheEntry('List')

此處removeCacheEntry方法清除的是列表組件的實(shí)例,'List' 值仍然在 組件的include里,下次重新進(jìn)入列表頁(yè)會(huì)重新加載列表組件,并且之后會(huì)繼續(xù)列表組件進(jìn)行緩存。

列表頁(yè)清除緩存的時(shí)機(jī)

進(jìn)入列表頁(yè)后清除緩存

在列表頁(yè)路由組件的beforeRouteEnter勾子中判斷是否是從其他頁(yè)面(Home)進(jìn)入的,是則清除緩存,不是則使用緩存。

defineOptions({
??name:?'List1',
??beforeRouteEnter?(to:?RouteRecordNormalized,?from:?RouteRecordNormalized)?{
????if?(from.name?===?'Home')?{
??????const?{?removeCacheEntry?}?=?useRouteCache()
??????removeCacheEntry('List1')
????}
??}
})

這種緩存方式有個(gè)不太友好的地方:當(dāng)從首頁(yè)進(jìn)入列表頁(yè),列表頁(yè)和詳情頁(yè)來回切換,列表頁(yè)是緩存的;但是在首頁(yè)和列表頁(yè)間用瀏覽器的前進(jìn)后退來切換時(shí),我們更多的是希望列表頁(yè)能保留緩存,就像在多頁(yè)面中瀏覽器前進(jìn)后退會(huì)緩存原頁(yè)面一樣的效果。但實(shí)際上,列表頁(yè)重新刷新了,這就需要使用另一種解決辦法,點(diǎn)擊鏈接時(shí)清除緩存清除緩存。

點(diǎn)擊鏈接跳轉(zhuǎn)前清除緩存

在首頁(yè)點(diǎn)擊跳轉(zhuǎn)列表頁(yè)前,在點(diǎn)擊事件的時(shí)候去清除列表頁(yè)緩存,這樣的話在首頁(yè)和列表頁(yè)用瀏覽器的前進(jìn)后退來回切換,列表頁(yè)都是緩存狀態(tài),只要當(dāng)重新點(diǎn)擊跳轉(zhuǎn)鏈接的時(shí)候,才重新加載列表頁(yè),滿足預(yù)期。

// 首頁(yè) Home.vue

<li>
  <router-link to="/list" @click="removeCacheBeforeEnter">列表頁(yè)</router-link>
</li>


<script setup lang="ts">
import useRouteCache from '@/hooks/useRouteCache'

defineOptions({
  name: 'Home'
})

const { removeCacheEntry } = useRouteCache()

// 進(jìn)入頁(yè)面前,先清除緩存實(shí)例
function removeCacheBeforeEnter () {
  removeCacheEntry('List')
}
</script>

狀態(tài)管理實(shí)現(xiàn)緩存

通過狀態(tài)管理庫(kù)存儲(chǔ)頁(yè)面的狀態(tài)和數(shù)據(jù)也能實(shí)現(xiàn)頁(yè)面緩存。此處狀態(tài)管理使用的是pinia。

首先使用pinia創(chuàng)建列表頁(yè)store:

import?{?defineStore?}?from?'pinia'

interface?Item?{
??id?:?number,
??content?:?string
}

const?useListStore?=?defineStore('list',?{
??//?推薦使用?完整類型推斷的箭頭函數(shù)
??state:?()?=>?{
????return?{
??????isRefresh:?true,
??????pageSize:?30,
??????currentPage:?1,
??????list:?[]?as?Item[],
??????curRow:?null?as?Item?|?null
????}
??},
??actions:?{
????setList?(data:?Item?[])?{
??????this.list?=?data
????},
????setCurRow?(data:?Item)?{
??????this.curRow?=?data
????},
????setIsRefresh?(data:?boolean)?{
??????this.isRefresh?=?data
????}
??}
})

export?default?useListStore

然后在列表頁(yè)中使用store:

<div>
  <el-page-header @back="goBack">
    <template #content>狀態(tài)管理實(shí)現(xiàn)列表頁(yè)緩存</template>
  </el-page-header>
  <el-table v-loading="loading" :data="tableData" border style="width: 100%; margin-top: 30px;">
    <el-table-column prop="id" label="id" />
    <el-table-column prop="content" label="內(nèi)容"/>
    <el-table-column label="操作">
      <template v-slot="{ row }">
        <el-link type="primary" @click="gotoDetail(row)">進(jìn)入詳情</el-link>
        <el-tag type="success" v-if="row.id === listStore.curRow?.id">剛點(diǎn)擊</el-tag>
      </template>
    </el-table-column>
  </el-table>
  <el-pagination
    v-model:currentPage="listStore.currentPage"
    :page-size="listStore.pageSize"
    layout="total, prev, pager, next"
    :total="listStore.list.length"
  />
</div>
  
<script setup lang="ts">
import useListStore from '@/store/listStore'
const listStore = useListStore()

...
</script>

通過beforeRouteEnter鉤子判斷是否從首頁(yè)進(jìn)來,是則通過?listStore.$reset()?來重置數(shù)據(jù),否則使用緩存的數(shù)據(jù)狀態(tài);之后根據(jù)?listStore.isRefresh?標(biāo)示判斷是否重新獲取列表數(shù)據(jù)。

defineOptions({
??beforeRouteEnter?(to:?RouteLocationNormalized,?from:?RouteLocationNormalized)?{
????if?(from.name?===?'Home')?{
??????const?listStore?=?useListStore()
??????listStore.$reset()
????}
??}
})

onBeforeMount(()?=>?{
??if?(!listStore.useCache)?{
????loading.value?=?true
????setTimeout(()?=>?{
??????listStore.setList(getData())
??????loading.value?=?false
????},?1000)
????listStore.useCache?=?true
??}
})

缺點(diǎn)

通過狀態(tài)管理去做緩存的話,需要將狀態(tài)數(shù)據(jù)都存在stroe里,狀態(tài)多起來的話,會(huì)有點(diǎn)繁瑣,而且狀態(tài)寫在store里肯定沒有寫在列表組件里來的直觀;狀態(tài)管理由于只做列表頁(yè)數(shù)據(jù)的緩存,對(duì)于一些非受控組件來說,組件內(nèi)部狀態(tài)改變是緩存不了的,這就導(dǎo)致頁(yè)面渲染后跟原來有差別,需要額外代碼操作。

頁(yè)面彈窗實(shí)現(xiàn)緩存

將詳情頁(yè)做成全屏彈窗,那么從列表頁(yè)進(jìn)入詳情頁(yè),就只是簡(jiǎn)單地打開詳情頁(yè)彈窗,將列表頁(yè)覆蓋,從而達(dá)到列表頁(yè) “緩存”的效果,而非真正的緩存。

這里還有一個(gè)問題,打開詳情頁(yè)之后,如果點(diǎn)后退,會(huì)返回到首頁(yè),實(shí)際上我們希望是返回列表頁(yè),這就需要給詳情彈窗加個(gè)歷史記錄,如列表頁(yè)地址為 '/list',打開詳情頁(yè)變?yōu)?'/list?id=1'。

彈窗組件實(shí)現(xiàn):

// PopupPage.vue

<template>
  <div class="popup-page" :class="[!dialogVisible && 'hidden']">
    <slot v-if="dialogVisible"></slot>
  </div>
</template>

<script setup lang="ts">
import { useLockscreen } from 'element-plus'
import { computed, defineProps, defineEmits } from 'vue'
import useHistoryPopup from './useHistoryPopup'

const props = defineProps({
  modelValue: {
    type: Boolean,
    default: false
  },
  // 路由記錄
  history: {
    type: Object
  },
  // 配置了history后,初次渲染時(shí),如果有url上有history參數(shù),則自動(dòng)打開彈窗
  auto: {
    type: Boolean,
    default: true
  },
  size: {
    type: String,
    default: '50%'
  },
  full: {
    type: Boolean,
    default: false
  }
})
const emit = defineEmits(
  ['update:modelValue', 'autoOpen', 'autoClose']
)

const dialogVisible = computed<boolean>({ // 控制彈窗顯示
  get () {
    return props.modelValue
  },
  set (val) {
    emit('update:modelValue', val)
  }
})

useLockscreen(dialogVisible)

useHistoryPopup({
  history: computed(() => props.history),
  auto: props.auto,
  dialogVisible: dialogVisible,
  onAutoOpen: () => emit('autoOpen'),
  onAutoClose: () => emit('autoClose')
})
</script>

<style lang='less'>
.popup-page {
  position: fixed;
  left: 0;
  right: 0;
  top: 0;
  bottom: 0;
  z-index: 100;
  overflow: auto;
  padding: 10px;
  background: #fff;
  
  &.hidden {
    display: none;
  }
}
</style>

彈窗組件調(diào)用:

<popup-page 
  v-model="visible" 
  full
  :history="{ id: id }">
  <Detail></Detail>
</popup-page>

缺點(diǎn)

彈窗實(shí)現(xiàn)頁(yè)面緩存,局限比較大,只能在列表頁(yè)和詳情頁(yè)中才有效,離開列表頁(yè)之后,緩存就會(huì)失效,比較合適一些簡(jiǎn)單緩存的場(chǎng)景。

父子路由實(shí)現(xiàn)緩存

該方案原理其實(shí)就是頁(yè)面彈窗,列表頁(yè)為父路由,詳情頁(yè)為子路由,從列表頁(yè)跳轉(zhuǎn)到詳情頁(yè)時(shí),顯示詳情頁(yè)字路由,且詳情頁(yè)全屏顯示,覆蓋住列表頁(yè)。

聲明父子路由:

{
??path:?'/list',
??name:?'list',
??component:?()?=>?import('./views/List.vue'),
??children:?[
????{
??????path:?'/detail',
??????name:?'detail',
??????component:?()?=>?import('./views/Detail.vue'),
????}
??]
}

列表頁(yè)代碼:文章來源地址http://www.zghlxwxcb.cn/news/detail-462163.html

//?列表頁(yè)
<template>
??<el-table?v-loading="loading"?:data="tableData"?border?style="width:?100%;?margin-top:?30px;">
????<el-table-column?prop="id"?label="id"?/>
????<el-table-column?prop="content"?label="內(nèi)容"/>
????<el-table-column?label="操作">
??????<template?v-slot="{?row?}">
????????<el-link?type="primary"?@click="gotoDetail(row)">進(jìn)入詳情</el-link>
????????<el-tag?type="success"?v-if="row.id?===?curRow?.id">剛點(diǎn)擊</el-tag>
??????</template>
????</el-table-column>
??</el-table>
??<el-pagination
????v-model:currentPage="currentPage"
????:page-size="pageSize"
????layout="total,?prev,?pager,?next"
????:total="list.length"
??/>
??
??<!--?詳情頁(yè)?-->
??<router-view?class="popyp-page"></router-view>
</template>

<style?lang='less'?scoped>
.popyp-page?{
??position:?fixed;
??top:?0;
??bottom:?0;
??left:?0;
??right:?0;
??z-index:?100;
??background:?#fff;
??overflow:?auto;
}
</style>

到了這里,關(guān)于Vue3 除了 keep-alive,還有哪些實(shí)現(xiàn)頁(yè)面緩存的方法的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點(diǎn)僅代表作者本人,不代表本站立場(chǎng)。本站僅提供信息存儲(chǔ)空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請(qǐng)注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實(shí)不符,請(qǐng)點(diǎn)擊違法舉報(bào)進(jìn)行投訴反饋,一經(jīng)查實(shí),立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費(fèi)用

相關(guān)文章

  • vue3中keep-alive的使用及結(jié)合transition使用

    在組件中使用(這里結(jié)合了 transition 內(nèi)置動(dòng)畫組件 ) 在router.js中配置 keepAlive 自定義屬性 VueCompilerError: expects exactly one child element or component. 不報(bào)錯(cuò),但 keep-alive 內(nèi)置組件的緩存沒有效果,onActivated 函數(shù)也不會(huì)執(zhí)行 vue中keep-alive的使用及詳解

    2024年02月11日
    瀏覽(19)
  • vue3,動(dòng)態(tài)引入組件,同時(shí)動(dòng)態(tài)設(shè)置組件的name,用于keep-alive緩存

    如果有兩個(gè)頁(yè)面邏輯大都相同,咱們想到的第一個(gè)肯定是寫一個(gè)組件,然后兩個(gè)路由都指向這個(gè)組件。 那如果現(xiàn)在多添加一個(gè)需求:兩個(gè)頁(yè)面在切換路由時(shí)都需要緩存數(shù)據(jù),并且兩個(gè)頁(yè)面的緩存數(shù)據(jù)要求獨(dú)立。 這個(gè)需求很簡(jiǎn)單:在router-view外層包裹一個(gè)keep-alive組件,指定緩

    2024年02月14日
    瀏覽(22)
  • vue3,vite開發(fā), 動(dòng)態(tài)引入組件,同時(shí)動(dòng)態(tài)設(shè)置組件的name,用于keep-alive緩存

    如果有兩個(gè)頁(yè)面邏輯大都相同,咱們想到的第一個(gè)肯定是寫一個(gè)組件,然后兩個(gè)路由都指向這個(gè)組件。 那如果現(xiàn)在多添加一個(gè)需求:兩個(gè)頁(yè)面在切換路由時(shí)都需要緩存數(shù)據(jù),并且兩個(gè)頁(yè)面的緩存數(shù)據(jù)要求獨(dú)立。 這個(gè)需求很簡(jiǎn)單:在router-view外層包裹一個(gè)keep-alive組件,指定緩

    2024年02月07日
    瀏覽(24)
  • Vue-組件緩存-keep-alive

    在Vue.js中,組件的復(fù)用和緩存是一個(gè)重要的優(yōu)化手段。當(dāng)我們頻繁切換組件時(shí),頻繁的銷毀和重建會(huì)帶來一定的性能開銷。為了解決這個(gè)問題,Vue提供了一個(gè)名為 keep-alive 的抽象組件。下面我們將深入探討 keep-alive 的工作原理、使用場(chǎng)景和最佳實(shí)踐。 keep-alive是什么 keep-aliv

    2024年01月16日
    瀏覽(44)
  • vue keep-alive 中的生命周期

    keep-alive 是 Vue 提供的一個(gè)內(nèi)置組件,用來對(duì)組件進(jìn)行緩存——在組件切換過程中將狀態(tài)保留在內(nèi)存中,防止重復(fù)渲染DOM。 如果為一個(gè)組件包裹了 keep-alive,那么它會(huì)多出兩個(gè)生命周期:deactivated、activated。同時(shí),beforeDestroy 和 destroyed 就不會(huì)再被觸發(fā)了,因?yàn)榻M件 不會(huì)被真正

    2024年02月12日
    瀏覽(23)
  • Vue2 - keep-alive 作用和原理

    Vue2 - keep-alive 作用和原理

    1,是一個(gè)內(nèi)部組件,用于緩存組件實(shí)例。在組件切換時(shí),不用重新創(chuàng)建而是使用緩存中的組件實(shí)例。 可以避免創(chuàng)建組件的開銷。 可以 保留組件狀態(tài) 。 2,有3個(gè)屬性: include 和 exclude 屬性,可以控制哪些組件進(jìn)入緩存。 max 屬性可以設(shè)置最大緩存數(shù),當(dāng)緩存的實(shí)例超過該數(shù)時(shí)

    2024年01月25日
    瀏覽(19)
  • Vue中的keep-alive緩存組件的理解

    keep-alive 是一個(gè)抽象組件,用于將其內(nèi)部的組件保留在內(nèi)存中,而不會(huì)重新渲染。這意味著當(dāng)組件在keep-alive 內(nèi)部切換時(shí),其狀態(tài)將被 保留 ,而不是被銷毀和重新創(chuàng)建。 ?????????keep-alive用來緩存不經(jīng)常變化的組件,以提高性能,當(dāng)我們?cè)诓煌酚芍g進(jìn)行切換或者是動(dòng)

    2024年01月18日
    瀏覽(37)
  • Vue keep-alive的使用和原理解析

    Vue keep-alive的使用和原理解析

    在當(dāng)今Web開發(fā)領(lǐng)域中,構(gòu)建交互性強(qiáng)、可復(fù)用且易于維護(hù)的用戶界面是至關(guān)重要的。而Vue.js作為一款現(xiàn)代化且流行的JavaScript框架,正是為了滿足這些需求而誕生。它采用了MVVM架構(gòu)模式,并通過數(shù)據(jù)驅(qū)動(dòng)和組件化的方式,使我們能夠更輕松地構(gòu)建出優(yōu)雅而高效的Web應(yīng)用程序。

    2024年01月16日
    瀏覽(24)
  • Vue.js 進(jìn)階技巧:keep-alive 緩存組件解析

    Vue.js 進(jìn)階技巧:keep-alive 緩存組件解析

    ?? 前端開發(fā)工程師、技術(shù)日更博主、已過CET6 ?? 阿珊和她的貓_ CSDN 博客專家、23年度博客之星前端領(lǐng)域TOP1 ?? ???高級(jí)專題作者、打造專欄《前端面試必備》 、《2024面試高頻手撕題》 ?? 藍(lán)橋云課 簽約作者、上架課程《Vue.js 和 Egg.js 開發(fā)企業(yè)級(jí)健康管理項(xiàng)目》、《帶你

    2024年03月11日
    瀏覽(20)
  • Vue路由組件的緩存keep-alive和include屬性

    功能:瀏覽器頁(yè)面在進(jìn)行切換時(shí),原有的路由組件會(huì)被銷毀。通過緩存可以保存被切換的路由組件。 例子:在頁(yè)面上填好的信息當(dāng)進(jìn)行頁(yè)面切換再轉(zhuǎn)回原來的頁(yè)面時(shí),原本信息被清空了需要重新填寫 功能:切換路由時(shí),保留被切換路由組件。 格式: keep-alive router-view/ keep-

    2024年02月06日
    瀏覽(25)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請(qǐng)作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包