從零開始實(shí)現(xiàn)vue-router珊拼,了解內(nèi)部機(jī)制

前端路由實(shí)現(xiàn)之 #hash

背景介紹

用了許多前端框架來做spa應(yīng)用食呻,比如說backbone,angular澎现,vue他們都有各自的路由系統(tǒng)仅胞,管理著前端的每一個(gè)頁面切換,想要理解其中路由的實(shí)現(xiàn)剑辫,最好的方法就是手動(dòng)實(shí)現(xiàn)一個(gè)干旧。
前端路由有2種實(shí)現(xiàn)方式,一種是html5推出的historyapi妹蔽,我們這里說的是另一種hash路由椎眯,就是常見的 # 號(hào)挠将,這種方式兼容性更好。

需求分析

我們這里只是簡單的實(shí)現(xiàn)一個(gè)路由輪子编整,基本的功能包含以下:

  1. 切換頁面
  2. 異步加載js
  3. 異步傳參

實(shí)現(xiàn)步驟

  1. 切換頁面:路由的最大作用就是切換頁面舔稀,以往后臺(tái)的路由是直接改變了頁面的url方式促使頁面刷新。但是前端路由通過 # 號(hào)不能刷新頁面掌测,只能通過 window 的監(jiān)聽事件 hashchange 來監(jiān)聽hash的變化内贮,然后捕獲到具體的hash值進(jìn)行操作

    //路由切換
    window.addEventListener('hashchange',function(){
        //do something 
        this.hashChange()
    })
    
  2. 注冊(cè)路由:我們需要把路由規(guī)則注冊(cè)到頁面,這樣頁面在切換的時(shí)候才會(huì)有不同的效果汞斧。

    //注冊(cè)函數(shù)
     map:function(path,callback){
       path = path.replace(/\s*/g,"");//過濾空格
       //在有回調(diào)夜郁,且回調(diào)是一個(gè)正確的函數(shù)的情況下進(jìn)行存儲(chǔ) 以 /name 為key的對(duì)象 {callback:xx}
       if(callback && Object.prototype.toString.call(callback) === '[object Function]' ){
           this.routers[path] ={
                callback:callback,//回調(diào)
                fn:null //存儲(chǔ)異步文件狀態(tài),用來記錄異步的js文件是否下載粘勒,下文有提及
            } 
        }else{
        //打印出錯(cuò)的堆棧信息
            console.trace('注冊(cè)'+path+'地址需要提供正確的的注冊(cè)回調(diào)')
        }
     }
     
     //調(diào)用方式
     map('/detail',function(transition){
      ...
    })
    
  3. 異步加載js:一般單頁面應(yīng)用為了性能優(yōu)化竞端,都會(huì)把各個(gè)頁面的文件拆分開,按需加載庙睡,所以路由里面要加入異步加載js文件的功能事富。異步加載我們就采用最簡單的原生方法,創(chuàng)建script標(biāo)簽埃撵,動(dòng)態(tài)引入js赵颅。

    var _body= document.getElementsByTagName('body')[0],
        scriptEle= document.createElement('script'); 
    scriptEle.type= 'text/javascript'; 
    scriptEle.src= xxx.js; 
    scriptEle.async = true;
    scriptEle.onload= function(callback){ 
        //為了避免重復(fù)引入js虽另,我們需要在這里記錄一下已經(jīng)加載過的文件暂刘,對(duì)應(yīng)的 fn需要賦值處理
        callback()
    } 
    _body.appendChild(scriptEle);   
    
  4. 參數(shù)傳遞:在我們動(dòng)態(tài)引入單獨(dú)模塊的js之后,我們可能需要給這個(gè)模塊傳遞一些單獨(dú)的參數(shù)捂刺。這里借鑒了一下jsonp的處理方式谣拣,我們把單獨(dú)模塊的js包裝成一個(gè)函數(shù),提供一個(gè)全局的回調(diào)方法族展,加載完成時(shí)候再調(diào)用回調(diào)函數(shù)森缠。

    SPA_RESOLVE_INIT = function(transition) { 
        document.getElementById("content").innerHTML = '<p style="color:#F8C545;">當(dāng)前異步渲染列表頁'+ JSON.stringify(transition) +'</p>'
        console.log("首頁回調(diào)" + JSON.stringify(transition))
    }
    

擴(kuò)展:以上我們已經(jīng)完成了基本功能,我們?cè)賹?duì)齊進(jìn)行擴(kuò)展仪缸,在頁面切換之前beforeEach和切換完成afterEach的時(shí)候增加2個(gè)方法進(jìn)行處理贵涵。思路是,注冊(cè)了這2個(gè)方法之后恰画,在切換之前就調(diào)用beforeEach宾茂,切換之后,需要等待下載js完成拴还,在onload里面進(jìn)行調(diào)用 afterEach

        //切換之前一些處理
        beforeEach:function(callback){
            if(Object.prototype.toString.call(callback) === '[object Function]'){
                this.beforeFun = callback;
            }else{
                console.trace('路由切換前鉤子函數(shù)不正確')
            }
        },
        //切換成功之后
        afterEach:function(callback){
            if(Object.prototype.toString.call(callback) === '[object Function]'){
                this.afterFun = callback;
            }else{
                console.trace('路由切換后回調(diào)函數(shù)不正確')
            }
        },

通過以上的思路分析跨晴,再加以整合,我們就完成了一個(gè)簡單的前端路由片林,并且可以加到頁面進(jìn)行實(shí)際的SPA開發(fā)端盆,不過還是非常簡陋怀骤。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市焕妙,隨后出現(xiàn)的幾起案子蒋伦,更是在濱河造成了極大的恐慌,老刑警劉巖焚鹊,帶你破解...
    沈念sama閱讀 218,122評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凉敲,死亡現(xiàn)場離奇詭異,居然都是意外死亡寺旺,警方通過查閱死者的電腦和手機(jī)爷抓,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來阻塑,“玉大人蓝撇,你說我怎么就攤上這事〕旅В” “怎么了渤昌?”我有些...
    開封第一講書人閱讀 164,491評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長走搁。 經(jīng)常有香客問我独柑,道長,這世上最難降的妖魔是什么私植? 我笑而不...
    開封第一講書人閱讀 58,636評(píng)論 1 293
  • 正文 為了忘掉前任忌栅,我火速辦了婚禮,結(jié)果婚禮上曲稼,老公的妹妹穿的比我還像新娘索绪。我一直安慰自己,他們只是感情好贫悄,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,676評(píng)論 6 392
  • 文/花漫 我一把揭開白布瑞驱。 她就那樣靜靜地躺著,像睡著了一般窄坦。 火紅的嫁衣襯著肌膚如雪唤反。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,541評(píng)論 1 305
  • 那天鸭津,我揣著相機(jī)與錄音彤侍,去河邊找鬼。 笑死曙博,一個(gè)胖子當(dāng)著我的面吹牛拥刻,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播父泳,決...
    沈念sama閱讀 40,292評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼般哼,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼吴汪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蒸眠,我...
    開封第一講書人閱讀 39,211評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤漾橙,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后楞卡,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體霜运,經(jīng)...
    沈念sama閱讀 45,655評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,846評(píng)論 3 336
  • 正文 我和宋清朗相戀三年蒋腮,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了淘捡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,965評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡池摧,死狀恐怖焦除,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情作彤,我是刑警寧澤膘魄,帶...
    沈念sama閱讀 35,684評(píng)論 5 347
  • 正文 年R本政府宣布,位于F島的核電站竭讳,受9級(jí)特大地震影響创葡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜绢慢,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,295評(píng)論 3 329
  • 文/蒙蒙 一灿渴、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧呐芥,春花似錦逻杖、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽闻伶。三九已至滨攻,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蓝翰,已是汗流浹背光绕。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評(píng)論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留畜份,地道東北人诞帐。 一個(gè)月前我還...
    沈念sama閱讀 48,126評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像爆雹,于是被迫代替她去往敵國和親停蕉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子愕鼓,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,914評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 概要 64學(xué)時(shí) 3.5學(xué)分 章節(jié)安排 電子商務(wù)網(wǎng)站概況 HTML5+CSS3 JavaScript Node 電子...
    阿啊阿吖丁閱讀 9,197評(píng)論 0 3
  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML標(biāo)準(zhǔn)慧起。 注意:講述HT...
    kismetajun閱讀 27,486評(píng)論 1 45
  • ## 框架和庫的區(qū)別?> 框架(framework):一套完整的軟件設(shè)計(jì)架構(gòu)和**解決方案**菇晃。> > 庫(lib...
    Rui_bdad閱讀 2,908評(píng)論 1 4
  • 1、JS的數(shù)據(jù)類型只有浮點(diǎn)型蚓挤,沒有整型磺送。null,underfined,boolean,number,string...
    6e5e50574d74閱讀 2,205評(píng)論 2 1
  • 一:什么是閉包?閉包的用處灿意? (1)閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)估灿。在本質(zhì)上,閉包就 是將函數(shù)內(nèi)部和函數(shù)外...
    xuguibin閱讀 9,607評(píng)論 1 52