全面解析JavaScript的Backbone.js框架中的Router路由
Backbone 中的 Router 充當路由的作用,控制 URL 的走向,當在 URL 中使用 # 標簽時生效。
定義 Router 至少需要一個 Router 和一個函數(shù)來映射特定的 URL,而且我們需要記住,在 Backbone 中,# 標簽后的任意字符都會被 Router 接收并解釋。
下面我們來定義一個 Router:
<script> var AppRouter = Backbone.Router.extend({ routes: { "*actions": "defaultRoute" // 匹配 http://example.com/#anything-here } }); // 實例化 Router var app_router = new AppRouter; app_router.on('route:defaultRoute', function(actions) { alert(actions); }) // 打開 Backbone 的歷史記錄 Backbone.history.start(); </script>
現(xiàn)在,我們就定義好了一個 Router 了,但此時 Router 并未匹配特定的 URL,接下來我們開始詳細講解 Router 是如何工作的。
動態(tài)路由選擇
Backbone 允許你定義帶有特定參數(shù)的 Router。例如,你可能希望通過一個特定的 id 接收一個 post,比如這樣一個 URL:"http://example.com/#/posts/12",一旦這個 Router 被激活,你就可以取得一個 id 為12的 post。接下來,我們就來定義這個 Router:
<script> var AppRouter = Backbone.Router.extend({ routes: { "posts/:id": "getPost", "*actions": "defaultRoute" //Backbone 會根據(jù)順序匹配路由 } }); // 實例化 Router var app_router = new AppRouter; app_router.on('route:getPost', function (id) { // 注意,參數(shù)通過這里進行傳遞 alert( "Get post number " + id ); }); app_router.on('route:defaultRoute', function (actions) { alert( actions ); }); // 打開 Backbone 的歷史記錄 Backbone.history.start(); </script>
匹配規(guī)則
Backbone 使用兩種形式的變量來設置 Router 的匹配規(guī)則。第一種是 :,它可以匹配 URL 中斜杠之間的任意參數(shù),另一種是 *,它用來匹配斜杠后面的所有部分。注意,由于第二種形式的模糊性大于第一種,所以它的匹配優(yōu)先級最低。
任一形式匹配的結果會以參數(shù)的形式傳遞到相關的函數(shù)中,第一種規(guī)則可能返回一個或多個參數(shù),第二種規(guī)則將整個匹配結果作為一個參數(shù)返回。
接下來,我們用實例來說明:
routes:{ "posts/:id": "getPost", // <a >Example</a> "download/*path": "downloadFile", // <a >Download</a> ":route/:action": "loadView", // <a >Load Route/Action View</a> }, app_router.on('route:getPost', function( id ){ alert(id); // 匹配后,傳遞過來的參數(shù)為 12 }); app_router.on('route:downloadFile', function( path ){ alert(path); // 匹配后,整個匹配結果作為一個參數(shù)返回,路徑為 user/images/hey.gif }); app_router.on('route:loadView', function( route, action ){ alert(route + "_" + action); // 匹配后,傳遞過來兩個參數(shù),此時會彈出 dashboard_graph });
你可能經(jīng)常聽說“路由器”這個詞,但它常常是指一種網(wǎng)絡設備,這種設備是網(wǎng)絡連接、數(shù)據(jù)傳輸?shù)膶Ш胶蜆屑~。而Backbone中的“路由器”功能與它類似,從上面的例子中你就能看出,它可以將不同的URL錨點導航到對應的Action方法。
(許多服務端Web框架中也提供了這樣的機制,但Backbone.Router更側重前端單頁應用的導航。)
Backbone的路由導航是由Backbone.Router和Backbone.History兩個類共同完成的:
- Router類用于定義和解析路由規(guī)則,并將URL映射到Action。
- History類用于監(jiān)聽URL的變化,和觸發(fā)Action方法。
我們一般不會直接實例化一個History,因為我們在第一次創(chuàng)建Router實例時,會自動創(chuàng)建一個History的單例對象,你可以通過Backbone.history來訪問這個對象。
要使用路由功能,首先我們需要定義一個Router類來聲明需要監(jiān)聽的URL規(guī)則和Action,在剛才的例子中,我們在定義時通過routes屬性來定義需要監(jiān)聽的URL列表,其中Key表示URL規(guī)則,Value表示當URL處于該規(guī)則時所執(zhí)行的Action方法。
Hash規(guī)則
URL規(guī)則表示當前URL中的Hash(錨點)片段,我們除了能在規(guī)則中指定一般的字符串外,還需要注意兩種特別的動態(tài)規(guī)則:
規(guī)則中以/(斜線)為分隔的一段字符串,在Router類內(nèi)部會被轉(zhuǎn)換為表達式([^\/]+),表示以/(斜線)開頭的多個字符,如果在這一段規(guī)則中設置了:(冒號),則表示URL中這一段字符串將被作為參數(shù)傳遞給Action。
例如我們設置了規(guī)則topic/:id,當錨點為#topic/1023時,1023將被作為參數(shù)id傳遞給Action,規(guī)則中的參數(shù)名(:id)一般會和Action方法的形參名稱相同,雖然Router并沒有這樣的限制,但使用相同的參數(shù)名更容易讓人理解。
規(guī)則中的*(星號)會在Router內(nèi)部被轉(zhuǎn)換為表達式(.*?),表示零個或多個任意字符,與:(冒號)規(guī)則相比,*(星號)沒有/(斜線)分隔的限制,就像我們在上面的例子中定義的*error規(guī)則一樣。
Router中的*(星號)規(guī)則在被轉(zhuǎn)換為正則表達式后使用非貪婪模式,因此你可以使用例如這樣的組合規(guī)則:*type/:id,它能匹配#hot/1023,同時會將hot和1023作為參數(shù)傳遞給Action方法。
上面介紹了規(guī)則的定義方式,這些規(guī)則都會對應一個Action方法名稱,該方法必須處于Router對象中。
在定義好Router類之后,我們需要實例化一個Router對象,并調(diào)用Backbone.history對象的start()方法,該方法會啟動對URL的監(jiān)聽。在History對象內(nèi)部,默認會通過onhashchange事件監(jiān)聽URL中Hash(錨點)的變化,對于不支持onhashchange事件的瀏覽器(例如IE6),History會通過setInterval心跳的方式監(jiān)聽。
pushState規(guī)則
Backbone.History還支持pushState方式的URL,pushState是HTML5提供的一種新特性,它能操作當前瀏覽器的URL(而不是僅僅改變錨點),同時不會導致頁面刷新,從而使單頁應用使用起來更像一套完整的流程。
要使用pushState特性,你需要先了解HTML5為該特性提供的一些方法和事件(這些方法都被定義在window.history對象中):
1.pushState():該方法可以將指定的URL添加一個新的history實體到瀏覽器歷史里
2.replaceState():該方法可以將當前的history實體替換為指定的URL
調(diào)用pushState()和replaceState()方法,僅僅是替換當前頁面的URL,而并不會真正轉(zhuǎn)到這個URL地址(當使用后退或前進按鈕時,也不會跳轉(zhuǎn)到該URL),我們可以通過onpopstate事件來監(jiān)聽這兩個方法引起的URL變化。
路由相關方法
1.route()方法
在設定好路由規(guī)則之后,如果需要動態(tài)調(diào)整,可以調(diào)用Router.route()方法來動態(tài)添加路由規(guī)則及Action方法,例如:
router.route('topic/:pageno/:pagesize', 'page', function(pageno, pagesize){ // todo }); 我們調(diào)用route()方法時,給定的規(guī)則不僅僅可以是字符串,也可以是一個正則表達式: router.route(/^topic/(.*?)/(.*?)$/, 'page', function(pageno, pagesize){ // todo });
2.navigate()方法
在前面的例子中,URL規(guī)則都是由我們手動輸入觸發(fā)的,在實際應用中,有時可能需要手動進行跳轉(zhuǎn)、導航,這時可以調(diào)用
Router.navigate()方法進行控制,例如: router.navigate('topic/1000', { trigger: true });
這段代碼將URL更改為http://localhost/index.html#topic/1000,并觸發(fā)了renderDetail方法。需要注意的是,我們在第二個參數(shù)傳入了trigger配置,該配置用于表示更改URL的同時是否觸發(fā)相應的Action方法。
3.stop()方法
還記得我們是通過Backbone.history.start()方法來啟動路由監(jiān)聽的,你也可以隨時調(diào)用Backbone.history.stop()方法來停止監(jiān)聽,例如:
router.route('topic/:pageno/:pagesize', 'page', function(pageno, pagesize) { Backbone.history.stop(); });
運行這段代碼,并訪問URL:http://localhost/index.html#topic/5/20,你會發(fā)現(xiàn)這個Action被執(zhí)行之后,監(jiān)聽已經(jīng)不再生效了。
相關文章
關于backbone url請求中參數(shù)帶有中文存入數(shù)據(jù)庫是亂碼的快速解決辦法
這篇文章主要介紹了關于backbone url請求中參數(shù)帶有中文存入數(shù)據(jù)庫是亂碼的快速解決辦法的相關資料,需要的朋友可以參考下2016-06-06JavaScript的Backbone.js框架入門學習指引
這篇文章主要介紹了JavaScript的Backbone.js框架入門學習指引, 其中特別講到了Backbone中的關鍵部分Router路由器,需要的朋友可以參考下2016-05-05詳解Backbone.js框架中的模型Model與其集合collection
這篇文章主要介紹了Backbone.js框架中的模型Model與其集合collection,Backbone擁有與傳統(tǒng)MVC框架相類似的Model與View結構,需要的朋友可以參考下2016-05-05深入解析Backbone.js框架的依賴庫Underscore.js的作用
這篇文章主要介紹了深入解析Backbone.js框架的依賴庫Underscore.js的作用,用過Node.js的朋友對Underscore一定不會陌生:)需要的朋友可以參考下2016-05-05講解JavaScript的Backbone.js框架的MVC結構設計理念
這篇文章主要介紹了JavaScript的Backbone.js框架的MVC結構設計理念,相比較于Angular.js,同樣為MVC結構的Backbone則顯得輕巧許多,需要的朋友可以參考下2016-02-02BackBone及其實例探究_動力節(jié)點Java學院整理
這篇文章主要介紹了BackBone及其實例探究,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07