一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

node.js|vue.js|jquery|angularjs|React|json|js教程|

服務器之家 - 編程語言 - JavaScript - vue.js - 詳解vue的hash跳轉(zhuǎn)原理

詳解vue的hash跳轉(zhuǎn)原理

2022-02-15 18:12mengyuhang4879 vue.js

這篇文章主要介紹了vue的hash跳轉(zhuǎn)原理,幫助大家更好的理解和學習使用vue框架,感興趣的朋友可以了解下

在new vueRouter的時候我們可以傳入一個mode屬性,他可以接收三個值:hash/history/abstract

詳解vue的hash跳轉(zhuǎn)原理

hash和history的區(qū)別

history的路徑更美觀一點 比如http://yoursite.com/user/id,history是基于pushState()來完成 URL 跳轉(zhuǎn)而無須重新加載頁面。 但是強制刷新還是會有問題(服務端來解決這個問題),所以history模式需要后端人員配合使用。

hash的路徑會帶有#,比如http://yoursite.com#/user/id

HashHistory

?
1
2
3
4
5
6
7
class VueRouter{
 constructor(options){
  this.matcher = createMatcher(options.routes || []);
//這里為了講解hash模式 所以就不進行判斷用戶傳進來的是哪種模式了
  this.history = new HashHistory(this);//this vue-router的實例
  }
}

源碼這里創(chuàng)建了一個基類我們這里和源碼統(tǒng)一,這個基類封裝了三種模式公用的方法和屬性,那么我們在這里創(chuàng)建一個HashHistory和基類History

?
1
2
3
4
5
6
7
8
9
10
11
12
13
import History from './base'
// hash路由
export default class HashHistory extends History{
 constructor(router){
  super(router); //繼承調(diào)用父類 等于call
 }
}
// 路由的基類
export default class History {
 constructor(router){
  this.router = router;
 }
}

如果是hash路由,打開網(wǎng)站如果沒有hash默認應該添加#/

?
1
2
3
4
5
6
7
8
9
10
11
12
13
import History from './base';
function ensureSlash(){
 if(window.location.hash){
  return
 }
 window.location.hash = '/'
}
export default class HashHistory extends History{
 constructor(router){
  super(router);
  ensureSlash(); // 確保有hash
 }
}

再看一下初始化的邏輯(上面的router.init函數(shù))

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
init(app){
  const history = this.history;
  // 初始化時,應該先拿到當前路徑,進行匹配邏輯
 
  // 讓路由系統(tǒng)過度到某個路徑
  const setupHashListener = ()=> {
   history.setupListener(); // 監(jiān)聽路徑變化
  }
  history.transitionTo( // 父類提供方法負責跳轉(zhuǎn)
   history.getCurrentLocation(), // 子類獲取對應的路徑
   // 跳轉(zhuǎn)成功后注冊路徑監(jiān)聽,為視圖更新做準備
   setupHashListener
  )
}

這里我們要分別實現(xiàn) transitionTo(基類方法)、 getCurrentLocation 、setupListener

getCurrentLocation實現(xiàn)

?
1
2
3
4
5
6
7
8
9
function getHash(){
 return window.location.hash.slice(1);
}
export default class HashHistory extends History{
 // ...
 getCurrentLocation(){
  return getHash();
 }
}

setupListener實現(xiàn)

?
1
2
3
4
5
6
7
8
9
export default class HashHistory extends History{
 // ...
 setupListener(){
  window.addEventListener('hashchange', ()=> {
   // 根據(jù)當前hash值 過度到對應路徑
   this.transitionTo(getHash());
  })
 }
}

TransitionTo實現(xiàn)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
export function createRoute(record, location) { // {path:'/',matched:[record,record]}
 let res = [];
 if (record) { // 如果有記錄
  while(record){
   res.unshift(record); // 就將當前記錄的父親放到前面
   record = record.parent
  }
 }
 return {
  ...location,
  matched: res
 }
}
export default class History {
 constructor(router) {
  this.router = router;
  // 根據(jù)記錄和路徑返回對象,稍后會用于router-view的匹配
  this.current = createRoute(null, {
   path: '/'
  })
 }
 // 核心邏輯
 transitionTo(location, onComplete) {
  // 去匹配路徑
  let route = this.router.match(location);
  // 相同路徑不必過渡
  if(
   location === route.path &&
   route.matched.length === this.current.matched.length){
   return
  }
  //更新路由并且下面會提到改變根實例上的_route屬性
  this.updateRoute(route)
  onComplete && onComplete();
 }
}
?
1
2
3
4
5
6
7
export default class VueRouter{
 // ...
 //做一個代理
 match(location){
  return this.matcher.match(location);
 }
}

macth方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
function match(location){ // 稍后根據(jù)路徑找到對應的記錄
 let record = pathMap[location]
 if (record) { // 根據(jù)記錄創(chuàng)建對應的路由
 //參數(shù):/about/a:{path:xx,component...},path:'/about/a'
  return createRoute(record,{
   path:location
  })
 }
 // 找不到則返回空匹配
 return createRoute(null, {
  path: location
 })
}

我們不難發(fā)現(xiàn)路徑變化時都會更改current屬性,我們可以把current屬性變成響應式的,每次current變化刷新視圖即可
在install方法中

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
install(Vue) {
 Vue.mixin({ // 給所有組件的生命周期都增加beforeCreate方法
  beforeCreate() {
   if (this.$options.router) {
   //調(diào)用Vue類中雙向數(shù)據(jù)綁定方法
   Vue.util.defineReactive(this,'_route',this._router.history.current);
   }
  }
 });
 // $route和$router方法 這兩個方法僅僅是vue中最常見的代理 僅僅是為了更加方便
 Object.defineProperty(Vue.prototype,'$route',{ // 每個實例都可以獲取到$route屬性
  get(){
   return this._routerRoot._route;//上面剛進行雙向數(shù)據(jù)綁定的
  }
 });
 Object.defineProperty(Vue.prototype,'$router',{ // 每個實例都可以獲取router實例
  get(){
   return this._routerRoot._router;
  }
 })
 }

切換路由每次初始化時都需要調(diào)用更新_route的方法,因為install的時候把_route進行雙向數(shù)據(jù)綁定,剛進來是沒有this._router.history.current的,通過發(fā)布訂閱方式來進行訂閱和更新操作;在init方法中增加監(jiān)聽函數(shù)

?
1
2
3
history.listen((route) => { // 需要更新_route屬性,出入一個函數(shù)
 app._route = route
});
?
1
2
3
4
5
6
7
8
9
10
11
12
13
export default class History {
 constructor(router) {
  // ...
  this.cb = null;
 }
 listen(cb){
  this.cb = cb; // 注冊函數(shù)
 }
 updateRoute(route){
  this.current =route;
  this.cb && this.cb(route); // 更新current后 更新_route屬性
 }
}

以上就是詳解vue的hash跳轉(zhuǎn)原理的詳細內(nèi)容,更多關于vue的hash跳轉(zhuǎn)原理的資料請關注服務器之家其它相關文章!

原文鏈接:https://segmentfault.com/a/1190000039369103

延伸 · 閱讀

精彩推薦
  • vue.js梳理一下vue中的生命周期

    梳理一下vue中的生命周期

    看過很多人講vue的生命周期,但總是被繞的云里霧里,尤其是自學的同學,可能js的基礎也不是太牢固,聽起來更是吃力,那我就已個人之淺見,以大白話...

    CRMEB技術團隊7992021-12-22
  • vue.jsVue2.x 項目性能優(yōu)化之代碼優(yōu)化的實現(xiàn)

    Vue2.x 項目性能優(yōu)化之代碼優(yōu)化的實現(xiàn)

    這篇文章主要介紹了Vue2.x 項目性能優(yōu)化之代碼優(yōu)化的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋...

    優(yōu)小U9632022-02-21
  • vue.js詳解vue 表單綁定與組件

    詳解vue 表單綁定與組件

    這篇文章主要介紹了vue 表單綁定與組件的相關資料,幫助大家更好的理解和學習使用vue框架,感興趣的朋友可以了解下...

    Latteitcjz6432022-02-12
  • vue.jsVue2.x-使用防抖以及節(jié)流的示例

    Vue2.x-使用防抖以及節(jié)流的示例

    這篇文章主要介紹了Vue2.x-使用防抖以及節(jié)流的示例,幫助大家更好的理解和學習使用vue框架,感興趣的朋友可以了解下...

    Kyara6372022-01-25
  • vue.jsVue多選列表組件深入詳解

    Vue多選列表組件深入詳解

    這篇文章主要介紹了Vue多選列表組件深入詳解,這個是vue的基本組件,有需要的同學可以研究下...

    yukiwu6752022-01-25
  • vue.jsVue項目中實現(xiàn)帶參跳轉(zhuǎn)功能

    Vue項目中實現(xiàn)帶參跳轉(zhuǎn)功能

    最近做了一個手機端系統(tǒng),其中遇到了父頁面需要攜帶參數(shù)跳轉(zhuǎn)至子頁面的問題,現(xiàn)已解決,下面分享一下實現(xiàn)過程,感興趣的朋友一起看看吧...

    YiluRen丶4302022-03-03
  • vue.js用vite搭建vue3應用的實現(xiàn)方法

    用vite搭建vue3應用的實現(xiàn)方法

    這篇文章主要介紹了用vite搭建vue3應用的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下...

    Asiter7912022-01-22
  • vue.jsVue中引入svg圖標的兩種方式

    Vue中引入svg圖標的兩種方式

    這篇文章主要給大家介紹了關于Vue中引入svg圖標的兩種方式,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的...

    十里不故夢10222021-12-31
主站蜘蛛池模板: 免费看男女做好爽好硬视频 | sese在线观看 | 皇上撞着太子妃的秘密小说 | 色姑娘色综合 | 国产一区二区三区在线看片 | 免费高清www动漫视频播放器 | 欧美日韩一区二区三区在线视频 | 国产精品2 | 国产专区视频在线观看 | 亚洲乱码一区二区三区国产精品 | 日韩毛片免费 | 百合互慰吃奶互揉漫画 | 日本老妇乱子伦中文视频 | 国产自在线观看 | 扒开双腿羞辱调教play视频 | 日本一区二区不卡久久入口 | 美女张开双腿让男人捅 | 桃色视频软件 | 俄罗斯妈妈235 | 欧美日韩国产另类一区二区三区 | 天色综合 | 欧美一级片在线免费观看 | 男人视频网站 | 免费国产之a视频 | 99ri国产在线观看 | 国产成人综合精品 | 亚洲欧美综合一区 | 免费成年人在线视频 | 99re8在这里只有精品23 | 国产精品va在线观看无 | a级黄色网 | 国产三级精品播放 | 精品99在线观看 | 免费观看无遮挡www的小视频 | 久久热r在线视频精品 | 国模丰满美女冰漪34d | 免费波多野结衣庭教师 | 日韩亚洲欧美理论片 | 国产精品成人麻豆专区 | 日本黄大片影院一区二区 | 日本视频免费在线观看 |