Vue中keep-alive 實現(xiàn)后退不刷新并保持滾動位置
什么是KeepAlive?
首先,我們要明確我們談的是TCP的 KeepAlive 還是HTTP的 Keep-Alive。TCP的KeepAlive和HTTP的Keep-Alive是完全不同的概念,不能混為一談。實際上HTTP的KeepAlive寫法是Keep-Alive,跟TCP的KeepAlive寫法上也有不同。
- TCP的keepalive是側(cè)重在保持客戶端和服務(wù)端的連接,一方會不定期發(fā)送心跳包給另一方,當(dāng)一方端掉的時候,沒有斷掉的定時發(fā)送幾次心跳包,如果間隔發(fā)送幾次,對方都返回的是RST,而不是ACK,那么就釋放當(dāng)前鏈接。設(shè)想一下,如果tcp層沒有keepalive的機(jī)制,一旦一方斷開連接卻沒有發(fā)送FIN給另外一方的話,那么另外一方會一直以為這個連接還是存活的,幾天,幾月。那么這對服務(wù)器資源的影響是很大的。
- HTTP的keep-alive一般我們都會帶上中間的橫杠,普通的http連接是客戶端連接上服務(wù)端,然后結(jié)束請求后,由客戶端或者服務(wù)端進(jìn)行http連接的關(guān)閉。下次再發(fā)送請求的時候,客戶端再發(fā)起一個連接,傳送數(shù)據(jù),關(guān)閉連接。這么個流程反復(fù)。但是一旦客戶端發(fā)送connection:keep-alive頭給服務(wù)端,且服務(wù)端也接受這個keep-alive的話,兩邊對上暗號,這個連接就可以復(fù)用了,一個http處理完之后,另外一個http數(shù)據(jù)直接從這個連接走了。減少新建和斷開TCP連接的消耗。
二者的作用簡單來說:
HTTP協(xié)議的Keep-Alive意圖在于短時間內(nèi)連接復(fù)用,希望可以短時間內(nèi)在同一個連接上進(jìn)行多次請求/響應(yīng)。
TCP的KeepAlive機(jī)制意圖在于?;?、心跳,檢測連接錯誤。當(dāng)一個TCP連接兩端長時間沒有數(shù)據(jù)傳輸時(通常默認(rèn)配置是2小時),發(fā)送keepalive探針,探測鏈接是否存活。
總之,記住HTTP的Keep-Alive和TCP的KeepAlive不是一回事。
tcp的keepalive是在ESTABLISH狀態(tài)的時候,雙方如何檢測連接的可用行。而http的keep-alive說的是如何避免進(jìn)行重復(fù)的TCP三次握手和四次揮手的環(huán)節(jié)。
正文開始。
vue可以通過<keep-alive>元素包裹組件,實現(xiàn)緩存,下次使用時不需要重新創(chuàng)建該組件。但存在一個問題:keep-alive包裹的組件中有滾動元素時,keep-alive不會儲存滾動位置。
實現(xiàn)后退不刷新主要依據(jù)keep-alive組件的activated和deactivated這兩個生命周期鉤子函數(shù)。
vue鉤子函數(shù)的執(zhí)行順序:
不使用keep-alive
beforeRouteEnter --> created --> mounted --> destroyed
使用keep-alive
初次進(jìn)入頁面,beforeRouteEnter --> created --> mounted --> activated --> deactivated
再次進(jìn)入緩存的頁面,只會觸發(fā)beforeRouteEnter -->activated --> deactivated。created和mounted不會再執(zhí)行。
其中,
activated在keep-alive組件激活時調(diào)用.
deactivated在keep-alive組件被停用時調(diào)用.
Demo 實現(xiàn)了后退不刷新,并且返回時滾動到上次瀏覽的深度。
該demo中,包含三個鏈接導(dǎo)航。
home --> pageA --> pageB --> pageC
依次前進(jìn),每次前進(jìn)到一個新頁面都需要獲取數(shù)據(jù),而按下后退鍵后,
從pageC返回到pageB,pageB不再獲取新數(shù)據(jù),而是使用之前緩存的數(shù)據(jù)。
從pageB返回到pageA時,pageA不再獲取新數(shù)據(jù),而是使用之前的數(shù)據(jù)。并且當(dāng)pageA存在滾動條時,返回時會滾動到上次瀏覽高度。
所以,pageA和pageB需要緩存,pageC不需要緩存。
//router.js
import Vue from 'vue';
import Router from 'vue-router';
Vue.use(Router);
const router = new Router({
mode: 'hash',
routes: [
{
path: '/',
name: 'home',
component: () =>
import('./views/Home.vue'),
meta: {
title: '首頁',
keepAlive: false //此組件不需要被緩存
}
},
{
path: '/pageA',
name: 'pageA',
component: () =>
import('./views/pageA.vue'),
meta: {
title: 'pageA',
keepAlive: true,
isBack: false
}
},
{
path: '/pageB',
name: 'pageB',
component: () =>
import('./views/pageB.vue'),
meta: {
title: 'pageB',
keepAlive: true,
isBack: false
}
},
{
path: '/pageC',
name: 'pageC',
component: () =>
import('./views/pageC.vue'),
meta: {
title: 'pageC',
keepAlive: false
}
}
]
});
export default router;
//pageA.vue
<template>
<div class="page-a">
<h1>pageA</h1>
<div>
<div class="item" v-for="item in items" @click="goPageB">
{{ item }}
</div>
</div>
<h1 @click="goPageB">go pageB</h1>
</div>
</template>
<script>
export default {
name: 'PageA',
data() {
return {
msg: "我是PageA頁面",
items: Array.from({length:50}, (v,k) => k),
data: "",
scrollTop: 0
};
},
beforeRouteEnter(to, from, next) {
if(from.name == 'pageB'){
to.meta.isBack = true;
}
next();
},
mounted() {
console.log('mounted....');
// this指向組件的實例,$el指向當(dāng)前組件的DOM元素
const $el = this.$el;
//滾動事件
$el.addEventListener("scroll", () => {
//記錄位置
this.scrollTop = $el.scrollTop;
});
},
activated() {
if(!this.$route.meta.isBack){
// 如果isBack是false,表明需要獲取新數(shù)據(jù),否則就不再請求,直接使用緩存的數(shù)據(jù)
this.getData();
} else {
//恢復(fù)滾動條高度
if(this.scrollTop) {
setTimeout(() => {
this.$el.scrollTop = this.scrollTop;
}, 100);
}
}
// 恢復(fù)成默認(rèn)的false,避免isBack一直是true
this.$route.meta.isBack = false;
},
methods: {
getData() {
// getData方法,模擬從后臺請求數(shù)據(jù)
this.data = "數(shù)據(jù)";
console.log('get data')
},
goPageB(){
this.$router.push({ path: "/pageB" });
},
back() {
this.$router.push({ path: "/" });
}
},
}
</script>
<style>
.page-a {
height: 100vh;
overflow-y: auto;
}
.item {
margin: 5px;
padding: 10px;
background: #ccc;
}
</style>
代碼請參考 鏈接 ;
后退不刷新還可以通過include實現(xiàn),可參考鏈接
總結(jié)
到此這篇關(guān)于Vue中keep-alive 實現(xiàn)后退不刷新并保持滾動位置的文章就介紹到這了,更多相關(guān)keep-alive 后退不刷新持滾動位置內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue3+ts重復(fù)參數(shù)提取成方法多處調(diào)用以及字段無值時不傳字段給后端問題
在進(jìn)行API開發(fā)時,優(yōu)化參數(shù)傳遞是一個重要的考量,傳統(tǒng)方法中,即使參數(shù)值為空,也會被包含在請求中發(fā)送給后端,這可能會導(dǎo)致不必要的數(shù)據(jù)處理,而優(yōu)化后的方法則只會傳遞那些實際有值的字段,從而提高數(shù)據(jù)傳輸?shù)挠行院秃蠖颂幚淼男?/div> 2024-10-10
Vue使用Element-UI實現(xiàn)分頁效果全過程
element-ui官網(wǎng)上有分頁實現(xiàn)的功能,簡單方便又好用,也有很多分頁的樣式,你可以根據(jù)需要去選擇自己想要的樣式,下面這篇文章主要給大家介紹了關(guān)于Vue使用Element-UI實現(xiàn)分頁效果的相關(guān)資料,需要的朋友可以參考下2023-04-04
用npm安裝vue和vue-cli,并使用webpack創(chuàng)建項目的方法
今天小編就為大家分享一篇用npm安裝vue和vue-cli,并使用webpack創(chuàng)建項目的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09
使用vite搭建ssr活動頁架構(gòu)的實現(xiàn)
本文主要介紹了使用vite搭建ssr活動頁架構(gòu),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07
詳解Vue中的render:?h?=>?h(App)示例代碼
這篇文章主要介紹了Vue中的render:?h?=>?h(App),本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-09-09最新評論

