前段路由实现思路

路由是什么

以路由器为例满足一对多的就叫做分发,分发请求的东西就是路由器
结合路由来看分发请求的对象就是路由

简单路由代码实现
<body>
    <a href="#1">go to 1</a>
    <a href="#2">go to 2</a>
    <a href="#3">go to 3</a>
    <a href="#4">go to 4</a>
    <div id="app"></div>
    <div id="div1" style="display: none">1</div>
    <div id="div2" style="display: none">2</div>
    <div id="div3" style="display: none">3</div>
    <div id="div4" style="display: none">4</div>
    <script src="src/index.js"></script>
  </body>
// 获取用户想去哪里
let number = window.location.hash.substring(1);
// 获取界面
let div = document.querySelector(`#div${number}`);
// 渲染界面
if (div) div.style.display = "block";
let app = document.querySelector("#app");
// 展示界面
app.appendChild(div);
// 监听路由的改变实现页面重新渲染
window.addEventListener("hashchange", e => {
  let number = window.location.hash.substring(1);
  let div = document.querySelector(`#div${number}`);
  if (div) div.style.display = "block";
  let app = document.querySelector("#app");
  app.children[0].style.display = "none";
  document.body.appendChild(app.children[0]);
  app.appendChild(div);
});

上面的#1、#2就叫做hash

路由分类:默认路由、保底路由、嵌套路由

默认路由:上面的页面如果我们不给#就会报错,这时候我们可以设置number= number || 1,这个一进去的页面也就是路由后面没有任何参数的

保底路由(404路由)
上面的代码如果我们后面的hash不是#1这些,而是#ccc就会报错,这时候我们就应该判断如果div不存在就让我们固定的一个div显示出来

if (div) {
  div.style.display = "block";
} else {
  div = document.querySelector("#div404");
  div.style.display = "block";
}

代码优化

function route() {
  let app = document.querySelector("#app");
  let number = window.location.hash.substring(1);
  number = number || 1;
  let div = document.querySelector(`#div${number}`);
  console.log(div);
  if (!div) {
    div = document.querySelector("#div404");
  }
  div.style.display = "block";
  if (app.children.length > 0) {
    app.children[0].style.display = "none";
    document.body.appendChild(app.children[0]);
  }
  app.appendChild(div);
}
route();
window.addEventListener("hashchange", e => {
  route();
});

路由表

上面的代码中我们的代码通过number来获取div对应的,比如number是1,我们就查找div1,那么我们凭什么要找div1,为什么不是one1那,所以我们需要通过一个表结构来自定义hash值跟div的对应关系

const div1 = document.createElement('div')
div1.innerHTML = '1'
const div2 = document.createElement('div')
div2.innerHTML = '2'
const div3 = document.createElement('div')
div3.innerHTML = '3'
const div4 = document.createElement('div')
div4.innerHTML = '4'
const routeTable = {
  '1': div1,
  '2': div2,
  '3': div3,
  '4': div4
}
function route() {
  let app = document.querySelector("#app");
  let number = window.location.hash.substring(1);
  number = number || 1;
  let div = routeTable[number.toString()]
  if (!div) {
    div = document.querySelector("#div404");
  }
  app.innerHTML = ''
  app.appendChild(div);
}
route()
window.addEventListener('hashchange', () => {
  route()
})

这样就能很明显的看出number和div之间的对应关系了

路由的三种模式

1. hash模式
适用场景:任何情况下都能做前端路由
缺点:SEO不友好(服务器收不到hash,也就是说浏览器是不会把#之后的内容发给服务器的
比如我们输入www.baidu.com/#aaa,当我们打开控制台查看请求的时候会发现请求的url依旧是www.baidu.com,但实际上我们的baidu.com/#aaabaidu.com里面的内容是不同的,所以对我们的seo很不友好

2. history模式
适用场景:后端将所有的前端路由都渲染到同一个页面(不包括404页面)
缺点:IE8以下不支持
比如:我不管输入baidu.com还是baidu.com/aaa都能访问首页

代码实现:
将之前的代码中的#改成/,然后通过window.location.pathname来获取它的路径

const div1 = document.createElement("div");
div1.innerHTML = "1";
const div2 = document.createElement("div");
div2.innerHTML = "2";
const div3 = document.createElement("div");
div3.innerHTML = "3";
const div4 = document.createElement("div");
div4.innerHTML = "4";
const routeTable = {
  "/1": div1,
  "/2": div2,
  "/3": div3,
  "/4": div4
};
function route() {
  let app = document.querySelector("#app");
  let number = window.location.pathname;
  if (number === '/') {
    number = '/1'
  }
  let div = routeTable[number.toString()];
  if (!div) {
    div = document.querySelector("#div404");
  }
  div.style.display = "block";
  app.innerHTML = "";
  app.appendChild(div);
}
route();
window.addEventListener("hashchange", () => {
  route();
});

问题:每次路径变的时候页面都会重新刷新渲染体验很差
解决方法使用history这个api

<a href="/1" class="link">go to 1</a>
    <a href="/2" class="link">go to 2</a>
    <a href="/3" class="link">go to 3</a>
    <a href="/4" class="link">go to 4</a>
    <div id="app"></div>

const allA = document.querySelectorAll("a.link");
for (let a of allA) {
  a.addEventListener("click", e => {
    e.preventDefault();
    const href = a.getAttribute("href");
    window.history.pushState(null, `page${href}`, href);
    onStateChange();
  });
}
function onStateChange() {
  route();
}
route();

上面的代码先获取所有的跳转页面对应的a标签,然后在点击他们的时候阻止默认的跳转行为然后通过属性拿到他们里面的路径,在通过history将对应的路径push进去,每次push完后重新渲染页面,就可以让页面不刷新了
完整代码https://codesandbox.io/embed/strange-wood-c54y7

3. memory模式
既不用hash又不用history,把路径存在用户看不见的地方(比如localStorage)
缺点:只对单机有效

function route() {
  let app = document.querySelector("#app");
  let number = window.localStorage.getItem("xxx");
  if (!number) {
    number = "/1";
  }
  let div = routeTable[number.toString()];
  if (!div) {
    div = document.querySelector("#div404");
  }
  div.style.display = "block";
  app.innerHTML = "";
  app.appendChild(div);
}
const allA = document.querySelectorAll("a.link");
for (let a of allA) {
  a.addEventListener("click", e => {
    e.preventDefault();
    const href = a.getAttribute("href");
    window.localStorage.setItem("xxx", href);
    onStateChange();
  });
}
function onStateChange() {
  route();
}
route();

三种模式的区别
hash和history模式都是把路径存在url上面,只要是不放在url上面的就是memory,memory因为没有对应的url,所以只对单机有效,不可分享;而hash和history可以通过url记录你的路由,所以是可分享的路由

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

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • “老大,你可以不给我工资,但,请给我一个帮你解决销售难题的机会。”我的语速很慢,但语气坚定。 年轻的女老板歪着头,...
    樽前邀月阅读 12,532评论 136 503
  • “人”即路人、朋友、亲人;即所有时间空间和你发生交集的那个“人”。我就是自己、所有能看到这段话的“人”。 世界真的...
    悟之道阅读 228评论 0 2
  • 90天打卡累计天数:48/90 1. 柯嘉12岁 运动目标:跳绳 ✘ 早睡早起:7:00起,10:00睡✔ 学习目...
    楼小楼de楼阅读 124评论 0 0