一文了解vue-router之hash模式和history模式


Posted in Javascript onMay 31, 2019

当前版本: 3.0.3

类目录: src/history/base.js

hash模式

即地址栏 URL 中的 # 符号(此 hash 不是密码学里的散列运算)。比如这个 URL: http://www.abc.com/#/hello ,hash 的值为 #/hello。它的特点在于:hash 虽然出现在 URL 中,但不会被包括在 HTTP 请求中,对后端完全没有影响,因此改变 hash 不会重新加载页面。

history模式

利用了 HTML5 History Interface 中新增的 pushState() 和 replaceState() 方法。(需要特定浏览器支持)这两个方法应用于浏览器的历史记录栈,在当前已有的 back、forward、go 的基础之上,它们提供了对历史记录进行修改的功能。只是当它们执行修改时,虽然改变了当前的 URL,但浏览器不会立即向后端发送请求。

HTML5History实现

使用window.addEventListener('popstate')监听浏览器滚动行为,然后判断配置是否有scrollBehavior, 有就调用handleScroll方法来处理

滚动行为: 使用前端路由,当切换到新路由时,想要页面滚到顶部,或者是保持原先的滚动位置,就像重新加载页面那样。 vue-router 能做到,而且更好,它让你可以自定义路由切换时页面如何滚动。

handleScroll

<!-- 等待页面渲染完才进行滚动的操作 -->
 router.app.$nextTick(() => {
   <!-- 初始化数据 -->
  const position = getScrollPosition()
  const shouldScroll = behavior.call(router, to, from, isPop ? position : null)

  if (!shouldScroll) {
   return
  }
  <!-- 判断是否是Promise,官网说支持异步 -->
  if (typeof shouldScroll.then === 'function') {
   shouldScroll.then(shouldScroll => {
    scrollToPosition((shouldScroll: any), position)
   }).catch(err => {
    if (process.env.NODE_ENV !== 'production') {
     assert(false, err.toString())
    }
   })
  } else {
   scrollToPosition(shouldScroll, position)
  }
 })

scrollToPosition

function scrollToPosition (shouldScroll, position) {
 const isObject = typeof shouldScroll === 'object'
 <!-- 对position进行初始化的操作 -->
 if (isObject && typeof shouldScroll.selector === 'string') {
  const el = document.querySelector(shouldScroll.selector)
  if (el) {
   let offset = shouldScroll.offset && typeof shouldScroll.offset === 'object' ? shouldScroll.offset : {}
   offset = normalizeOffset(offset)
   position = getElementPosition(el, offset)
  } else if (isValidPosition(shouldScroll)) {
   position = normalizePosition(shouldScroll)
  }
 } else if (isObject && isValidPosition(shouldScroll)) {
  position = normalizePosition(shouldScroll)
 }
  使用window.scrollTo来进行滚动处理
 if (position) {
  window.scrollTo(position.x, position.y)
 }
}

push

push操作也是 HTML5History模式下的一个比较关键的方法,他使用pushState来进行跳转操作,然后handleScroll来进行滚动\

export function pushState (url?: string, replace?: boolean) {
  <!-- 保存当前页面的滚动位置 -->
 saveScrollPosition()
 // try...catch the pushState call to get around Safari
 // DOM Exception 18 where it limits to 100 pushState calls
 const history = window.history
 try {
   <!-- 判断是哪种操作动作 -->
  if (replace) {
   history.replaceState({ key: _key }, '', url)
  } else {
   _key = genKey()
   history.pushState({ key: _key }, '', url)
  }
 } catch (e) {
  window.location[replace ? 'replace' : 'assign'](url)
 }
}

HashHistory实现

对于HashHistory的实现,和HTML5History的区别是在于Listener的方式和跳转的方式

Listener的方式这里是使用了hashchange,但是如果需要滚动行为就会去监听popstate

window.addEventListener(supportsPushState ? 'popstate' : 'hashchange')

跳转的方式会判断是否需要滚动,不需要就直接使用window.location.hash

function pushHash (path) {
 if (supportsPushState) {
  pushState(getUrl(path))
 } else {
  window.location.hash = path
 }
}

总结

以上所述是小编给大家介绍的vue-router之hash模式和history模式,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Javascript 相关文章推荐
javascript 表单规则集合对象
Jul 21 Javascript
简易js代码实现计算器操作
Apr 15 Javascript
点击页面其它地方隐藏该div的两种思路
Nov 18 Javascript
jquery 判断滚动条到达了底部和顶端的方法
Apr 02 Javascript
jQuery实现固定在网页顶部的菜单效果代码
Sep 02 Javascript
微信js-sdk上传与下载图片接口用法示例
Oct 12 Javascript
js + css实现标签内容切换功能(实例讲解)
Oct 09 Javascript
echarts统计x轴区间的数值实例代码详解
Jul 07 Javascript
layui动态加载多表头的实例
Sep 05 Javascript
JS页面获取 session 值,作用域和闭包学习笔记
Oct 16 Javascript
小程序实现图片移动缩放效果
May 26 Javascript
原生JS实现京东查看商品点击放大
Dec 21 Javascript
vue App.vue中的公共组件改变值触发其他组件或.vue页面监听
May 31 #Javascript
微信小程序环境下将文件上传到OSS的方法步骤
May 31 #Javascript
Vue Router history模式的配置方法及其原理
May 30 #Javascript
vue-cli3+ts+webpack实现多入口多出口功能
May 30 #Javascript
详解Vue项目引入CreateJS的方法(亲测可用)
May 30 #Javascript
了解JavaScript函数中的默认参数
May 30 #Javascript
Element-ui中元素滚动时el-option超出元素区域的问题
May 30 #Javascript
You might like
PHP5 面向对象程序设计
2008/02/13 PHP
执行、获取远程代码返回:file_get_contents 超时处理的问题详解
2013/06/25 PHP
ThinkPHP 404页面的设置方法
2015/01/14 PHP
替换php字符串中的单引号为双引号的方法
2017/02/16 PHP
php swoft框架实例用法
2020/12/22 PHP
Javascript 判断 object 的特定类转载
2007/02/01 Javascript
一个简单的js渐显(fadeIn)渐隐(fadeOut)类
2010/06/19 Javascript
jQuery + Flex 通过拖拽方式动态改变图片的代码
2011/08/03 Javascript
javascript jscroll模拟html元素滚动条
2012/12/18 Javascript
js控制滚动条缓慢滚动到顶部实现代码
2013/03/20 Javascript
js中AppendChild与insertBefore的用法详细解析
2013/12/16 Javascript
JavaScript字符串对象slice方法入门实例(用于字符串截取)
2014/10/16 Javascript
JavaScript中实现继承的三种方式和实例
2015/01/29 Javascript
JQuery遍历DOM节点的方法
2015/06/11 Javascript
快速学习JavaScript的6个思维技巧
2015/10/13 Javascript
深入解析JavaScript中的arguments对象
2016/06/12 Javascript
vue的props实现子组件随父组件一起变化
2016/10/27 Javascript
详解百度百科目录导航树小插件
2017/01/08 Javascript
javascript 跨域问题以及解决办法
2017/07/17 Javascript
IntersectionObserver实现图片懒加载的示例
2017/09/29 Javascript
javascript自定义日期比较函数用法示例
2019/07/22 Javascript
layui 实现table翻页滚动条位置保持不变的例子
2019/09/05 Javascript
微信JSSDK实现打开摄像头拍照再将相片保存到服务器
2019/11/15 Javascript
[01:10]DOTA2英雄背景故事第四期之混沌法则混沌骑士
2020/07/16 DOTA
一步步教你用Python实现2048小游戏
2017/01/19 Python
Django的性能优化实现解析
2019/07/30 Python
python3.7 openpyxl 删除指定一列或者一行的代码
2019/10/08 Python
实列教程 一款基于jquery和css3的响应式二级导航菜单
2014/11/13 HTML / CSS
巴西食品补充剂在线零售商:Músculos na Web
2017/08/07 全球购物
娇韵诗Clarins意大利官方网站:法国天然护肤品牌
2020/03/11 全球购物
澳大利亚在线床零售商:Bedworks
2020/09/01 全球购物
行政部主管岗位职责
2013/12/28 职场文书
大学生毕业自我鉴定范文
2014/02/03 职场文书
出资证明书范本(标准版)
2014/09/24 职场文书
世界文化遗产导游词
2019/08/07 职场文书
小型企业的绩效考核制度模板
2019/11/21 职场文书