前端实现路由

前端实现路由有两种方式,如下

History Api

自从H5开始,js的history就引入了<strong>history.pushState</strong>和<strong>history.replaceState</strong>两个方法,这两个允许添加和修改history实体,并可以和<strong>window.onpoState</strong>事件一起运作。

history.pushState(state,title,url)
history.replaceState(state,title,url)

它们基本相同,唯一不同的就是:pushStage 增加一条新的历史记录,而replaceState 替换当前的历史记录

state:state对象是一个JavaScript对象,它关系到由pushState()方法创建出来的新的history实体
title: 现在并无太大的意义
url:用来传递给新history实体的url,浏览器并不会真正的去加载这个url,但在用户重启浏览器时,有可能会加载这个url,url可以是绝对地址(必须是同意域名)或是相对地址(拼接在url后面)

下面看实例(注意地址栏url的变化):

<!DOCTYPE HTML>
<!-- this starts off as http://example.com/line?x=5 -->
<html>
<title>Line Game - 5</title>
<p>You are at coordinate <span id="coord">5</span> on the line.</p>
<p>
 <a href="?x=6" onclick="go(1); return false;">Advance to 6</a> or
 <a href="?x=4" onclick="go(-1); return false;">retreat to 4</a>?
</p>
<script>
 var currentPage = 5; // prefilled by server!!!!
 function go(d) {
    setupPage(currentPage + d);
     history.pushState(currentPage, null, '?x=' + currentPage);
 }
 onpopstate = function(event) {
     setupPage(event.state);
 }
 function setupPage(page) {
     currentPage = page;
     document.title = 'Line Game - ' + currentPage;
     document.getElementById('coord').textContent = currentPage;
     document.links[0].href = '?x=' + (currentPage+1);
     document.links[0].textContent = 'Advance to ' + (currentPage+1);
     document.links[1].href = '?x=' + (currentPage-1);
     document.links[1].textContent = 'retreat to ' + (currentPage-1);
 }
</script>
<body>
</body  >
</html>

hash

在一些网站中,我们经常会看到一些url带#,如标题之间切换、回到顶部等,在这里我们称为<strong>锚点</strong>,而在路由中,我们称为<strong>hash</strong>,并监听<strong>onhashchange</strong>事件。hash经常应用于大型框架路由中。

运行如下代码:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="keywords" content="javascript" />
<meta name="description" content="geg" />
<title>演示:Javascript实现前端简单路由</title>
<link rel="stylesheet" href="//cdn.bootcss.com/bootstrap/3.3.5/css/bootstrap.min.css">

<style>
.text-right li{padding: 10px}
#result{height: 200px; line-height: 200px; font-size: 2rem; text-align: center; color:#fff;}
</style>
</head>
<body>
<div class="container">
    <header>
        <div class="row">
            <div class="col-md-3 col-xs-12"><h1 class="logo"><a href="http://www.helloweba.com" title="返回helloweba首页">helloweba</a></h1></div>
            <div class="col-md-9 text-right"></div>
        </div>
    </header>
    <div class="row main" style="min-height:500px">
        <div class="col-md-12">
            <h2 class="top_title"><span class="glyphicon glyphicon-menu-left" aria-hidden="true"></span><a href="http://www.helloweba.com/view-blog-385.html">Javascript实现前端简单路由</a></h2>
            
            <div class="row" style="margin-top:30px">
                <div class="col-md-3">
                    <ul class="text-right"> 
                        <li><a href="#/">首页</a></li> 
                        <li><a href="#/product">产品</a></li> 
                        <li><a href="#/server">服务</a></li> 
                    </ul>
                </div>
                <div class="col-md-7">
                    <div id="result"></div>
                </div>
            </div>
        </div>
    </div>

    <footer>
        <p>Powered by helloweba.com  允许转载、修改和使用本站的DEMO,但请注明出处:<a href="http://www.helloweba.com">www.helloweba.com</a></p>
    </footer>
</div>
<script type="text/javascript">
function Router(){
    
    this.routes = {};
    this.curUrl = '';

    this.route = function(path, callback){
        this.routes[path] = callback || function(){};
    };

    this.refresh = function(){
        this.curUrl = location.hash.slice(1) || '/';
        this.routes[this.curUrl]();
    };

    this.init = function(){
        window.addEventListener('load', this.refresh.bind(this), false);
        window.addEventListener('hashchange', this.refresh.bind(this), false);
    }

}

var R = new Router();
R.init();
var res = document.getElementById('result');
 R.route('/', function() {
    res.style.background = 'blue';
    res.innerHTML = '这是首页';
 });
 R.route('/product', function() {
    res.style.background = 'orange';
    res.innerHTML = '这是产品页';
 });
 R.route('/server', function() {
    res.style.background = 'black';
    res.innerHTML = '这是服务页';
 });
</script>
</body>
</html>

R::init()注册了监听事件(页面加载结束和hash变化)
R::router 存储路由更新时的回调到回调数组routes中,回调函数将负责对页面的更新。
refresh 执行当前url对应的回调函数,更新页面。

以上只是对路由的简单操作,实际中可以对hash加上正则的限制,同时增加ajax异步请求

参考来源 :http://www.helloweba.com/view-blog-385.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • /*Returns thenumberof entries in the joint sessionhistory...
    heachou阅读 636评论 0 1
  • 什么是前端路由路由,引导、指路之意。 譬如我们熟知的路由器,蹦跶在网络层的数据包转发设备,在网络中也是扮演着指路明...
    蓝浅蓝深阅读 754评论 0 5
  • 本文总结自:Web开发中 前端路由 实现的几种方式和适用场景HTML 5 History API的”前生今世”从零...
    始悔不悟阅读 1,659评论 0 2
  • 初恋这件小事(年月日) 绿萝的最后一枝叶坚强地活着,看到旁边新爆出的嫩芽了吗,似乎充满着希望。 但我今天要对它说的...
    木筱茜阅读 560评论 2 1
  • 今天是自己坚持写文的第九天了,虽然时间很短,也短的可怜,这也足以让我小小的开心一次了。因为,在我二十年的生活里,从...
    与梦阅读 329评论 0 0