HTML5实现无刷新修改URL的方法


Posted in HTML / CSS onNovember 14, 2019

前言

今天在做一个vue的搜索功能,需要从搜索结果页面跳转到细节页面,然后点击返回还能返回到刚刚的结果页面,如果只用window.history.go(-1)当然会重新刷新搜索页面,当然是不行的。

我尝试了俩种方式来修改url:

  • window.location.href,拼接一个搜索的key值,点击搜索的同时,刷新了页面,url改变了,功能是实现了,可是bug来了...,搜索页面闪烁后才进入结果页,而结合不跳转页面就不会发生闪烁的情况,所以当页面刷新时,vue实例都会被重新加载。
  • window.location.hash,url中#为网页中的一个位置,当读取此url时,页面会自动滚动到锚点处。这种方法不会向服务器发送请求,也不会重载页面。可读写。读取时,可以用来判断网页状态是否改变;写入时,则会在不重载网页的前提下,创造一条访问历史记录。

新增API

百度后发现HTML5为window.history对象引入了两个新的api是history.pushState()history.replaceState()方法,它们分别可以添加和修改历史记录条目。这些方法通常与window.onpopstate事件配合使用。两种api都能改变当前的url,不同的是,pushState在浏览器中创建一条新的历史记录,而history.replaceState仅仅替换当前地址为指定url。

HTML5 新增的历史记录 API 均可以实现无刷新更改地址栏链接,配合 AJAX 完全可以做到无刷新跳转。他们的作用非常大,可以做到改变网址却不需要刷新页面,这个特性后来用到了单页面应用中比如:vue-router,react-router-dom里面。那么下面介绍新API的使用方法

pushState方法

上面提到的这套 API 提供一种「人为操纵」浏览器历史记录的方法。

我们可以将浏览器历史记录可以看作一个「栈」。栈是一种后进先出的结构,可以把它想象成一摞盘子,用户每点开一个新网页,都会在上面加一个新盘子,叫「入栈」。用户每次点击「后退」按钮都会取走最上面的那个盘子,叫做「出栈」。而每次浏览器显示的自然是最顶端的盘子的内容。

语法:window.history.pushState(state, title, url);

执行pushState函数之后,会往浏览器的历史记录中创建一条新记录,同时改变地址栏的地址内容。它可以接收三个参数,按语法顺序分别为:

  • 一个state对象或者字符串,用于描述新记录的一些特性。这个参数会被一并添加到历史记录中,以供以后使用。这个参数是开发者根据自己的需要自由给出的。该值是后期可以通过history.state;获取当前页面的参数,
  • 一个字符串,代表新页面的标题。当前基本上所有浏览器都会忽略这个参数。
  • 一个字符串,代表新页面的相对地址。(必须与当前页面处在同一个域。)

简单例子:假设当前页面为renfei.org/,打开控制台执行下面的 JavaScript 语句:

window.history.pushState({id: 1,name: "profile1"}, "My Profile", "/profile/");  //第一二参数可忽略设置为null

之后,地址栏的地址就会变成renfei.org/profile/,但同时浏览器不会刷新页面,甚至不会检测目标页面是否存在。

replaceState方法

有时,你希望不添加一个新记录,而是替换当前的记录,俩者区别在于 replaceState() 是修改了当前的历史记录项而不是新建一个。

replaceState和pushState原理一样使用也一样,最常用的方法:

window.history.replaceState({id: 1,name: "profile"},'下载','download?id=1');

特点:replaceState不会加入到历史记录里面,用history.go(-1)会跳过当前页面相当于是history.go(-2)。

popstate事件

当页面加载时,它可能会有一个非空的state对象。这可能发生在当页面设置一个state对象(使用pushState或者replaceState)之后用户重启了浏览器。当页面重新加载,页面将收到onload事件,但不会有popstate事件。然而,如果你读取history.state属性,将在popstate事件发生后得到这个state对象

var currentState = history.state; //得到当前页面设置的history.pushState的state对象参数值

调用 history.pushState() 或者 history.replaceState() 不会触发 popstate 事件。 popstate事件只会在浏览器某些行为下触发, 比如点击后退、前进按钮(或者在JavaScript中调用history.back()、history.forward()、history.go()方法)。

例子:假设当前页面地址是http://example.com/index.html

window.onpopstate = function(event) {
  alert("location: " + document.location.href + ", state: " + JSON.stringify(event.state));//拿到history.state对象值
};
//绑定事件处理函数. 
history.pushState({page: 1}, "title 1", "?page=100");    //添加并激活一个历史记录条目 http://example.com/index.html?page=100,条目索引为1    //history.state为{page: 1}
history.pushState({page: 2}, "title 2", "?page=200");    //添加并激活一个历史记录条目 http://example.com/index.html?page=200,条目索引为2
history.replaceState({page: 3}, "title 3", "?page=300"); //修改当前所在页面激活的历史记录条目 http://ex..?page=200 变为 http://ex..?page=300,条目索引为3

history.back(); // 弹出 "location: http://example.com/index.html?page=100, state: {"page":1}"
history.back(); // 弹出 "location: http://example.com/index.html, state: null
history.go(2);  // 弹出 "location: http://example.com/index.html?page=300, state: {"page":3}

pushState方法与设置hash值的区别

在某种意义上,调用 pushState() 与 设置 window.location = "#foo" 类似,二者都会在当前页面创建并激活新的历史记录。但 pushState() 具有如下几条优点:

新的 URL 可以是与当前URL同源的任意URL 。而设置 window.location 仅当你只修改了哈希值时才保持同一个文件。如果需要,可以不必改变URL就能创建一条历史记录。而设置 window.location.hash = "#foo";只有在当前哈希不是 #foo 的情况下, 才会创建一个新的历史记录项。我们可以为新的历史记录项关联任意数据。而基于哈希值的方式,则必须将所有相关数据编码到一个短字符串里。

需要注意的是通过history.pushState修改网页的URL地址,在配合相关代码显示隐藏相应界面便可以实现单页面多界面相互操作。该方法比直接访问URL地址速度快,执行效率高,UI体验好,但会增加页面的复杂性及耦合性,要视实际情况而定,一般都用在dialog弹出框上。

应用:全站 AJAX,并使浏览器能够抓取 AJAX 页面

这个可以干啥用?一个比较常用的场景就是,配合 AJAX。

假设一个页面左侧是若干导航链接,右侧是内容,同时导航时只有右侧的内容需要更新,那么刷新整个页面无疑是浪费的。这时我们可以使用 AJAX 来拉取右面的数据。但是如果仅仅这样,地址栏是不会改变的,用户无法前进、后退,也无法收藏当前页面或者把当前页面分享给他人;搜索引擎抓取也有困难。这时,就可以使用 HTML5 的 History API 来解决这个问题。

思路:首先绑定click事件。当用户点击一个链接时,通过preventDefault函数防止默认的行为(页面跳转),同时读取链接的地址(如果有 jQuery,可以写成$(this).attr('href')),把这个地址通过pushState塞入浏览器历史记录中,再利用 AJAX 技术拉取(如果有 jQuery,可以使用$.get方法)这个地址中真正的内容,同时替换当前网页的内容。

为了处理用户前进、后退,我们监听popstate事件。当用户点击前进或后退按钮时,浏览器地址自动被转换成相应的地址,同时popstate事件发生。在事件处理函数中,我们根据当前的地址抓取相应的内容,然后利用 AJAX 拉取这个地址的真正内容,呈现,即可。

最后,整个过程是不会改变页面标题的,可以通过直接对document.title赋值来更改页面标题。

总结

以上所述是小编给大家介绍的HTML5实现无刷新修改URL的方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

HTML / CSS 相关文章推荐
CSS3 @font-face属性使用指南
Dec 12 HTML / CSS
利用HTML5+CSS3实现3D转换效果实例详解
May 02 HTML / CSS
利用css3径向渐变做一张优惠券的示例
Mar 22 HTML / CSS
css3动画鼠标放上图片逐渐变大鼠标离开图片逐渐缩小效果
Jan 27 HTML / CSS
从一次项目重构说起CSS3自定义变量在项目的使用方法
Mar 01 HTML / CSS
分享一个页面平滑滚动小技巧(推荐)
Oct 23 HTML / CSS
突袭HTML5之Javascript API扩展3—本地存储全新体验
Jan 31 HTML / CSS
html5基础标签(html5视频标签 html5新标签用法)
Dec 30 HTML / CSS
HTML5 LocalStorage 本地存储刷新值还在
Mar 10 HTML / CSS
html5使用Drag事件编辑器拖拽上传图片的示例代码
Aug 22 HTML / CSS
html form表单基础入门案例讲解
Jul 15 HTML / CSS
HTML5 新增内容和 API详解
Nov 17 HTML / CSS
iframe跨域的几种常用方法
Nov 11 #HTML / CSS
Canvas环形饼图与手势控制的实现代码
Nov 08 #HTML / CSS
Html5监听手机摇一摇事件的实现
Nov 07 #HTML / CSS
解决HTML5中的audio在手机端和微信端的不能自动播放问题
Nov 04 #HTML / CSS
详解使用postMessage解决iframe跨域通信问题
Nov 01 #HTML / CSS
跨域修改iframe页面内容详解
Oct 31 #HTML / CSS
html如何对span设置宽度
Oct 30 #HTML / CSS
You might like
php将12小时制转换成24小时制的方法
2015/03/31 PHP
php基于CodeIgniter实现图片上传、剪切功能
2016/05/14 PHP
Thinkphp实现短信验证注册功能
2016/10/18 PHP
PHP Post获取不到非表单数据的问题解决办法
2018/02/27 PHP
thinkphp5.1 文件引入路径问题及注意事项
2018/06/13 PHP
intro.js 页面引导简单用法 分享
2013/08/06 Javascript
javascript向后台传送相同属性的参数即数组参数
2014/02/17 Javascript
Bootstrap按钮组件详解
2016/04/26 Javascript
避免jQuery名字冲突 noConflict()方法
2016/07/30 Javascript
Reactjs实现通用分页组件的实例代码
2017/01/19 Javascript
ajax 提交数据到后台jsp页面及页面跳转问题
2017/01/19 Javascript
web前端开发中常见的多列布局解决方案整理(一定要看)
2017/10/15 Javascript
Angular请求防抖处理第一次请求失效问题
2019/05/17 Javascript
JS获取当前时间的年月日时分秒及时间的格式化的方法
2019/12/18 Javascript
vant组件中 dialog的确认按钮的回调事件操作
2020/11/04 Javascript
8个非常实用的Vue自定义指令
2020/12/15 Vue.js
python益智游戏计算汉诺塔问题示例
2014/03/05 Python
Python实现LRU算法的2种方法
2015/06/24 Python
简单谈谈python的反射机制
2016/06/28 Python
深入解析Python中的上下文管理器
2016/06/28 Python
Python实现求解括号匹配问题的方法
2018/04/17 Python
Python实现手写一个类似django的web框架示例
2018/07/20 Python
树莓派4B+opencv4+python 打开摄像头的实现方法
2019/10/18 Python
详解Python中第三方库Faker
2020/09/25 Python
微信浏览器左上角返回按钮拦截功能
2017/11/21 HTML / CSS
Hotels.com台湾:饭店订房网
2017/09/06 全球购物
迪士尼西班牙官方网上商店:ShopDisney西班牙
2020/02/02 全球购物
意大利单身交友网站:Meetic
2020/07/12 全球购物
C语言编程题
2015/03/09 面试题
实习教师自我鉴定
2013/09/27 职场文书
医科大学生的自我评价
2013/12/04 职场文书
餐饮服务食品安全责任书
2014/07/25 职场文书
2014年十一国庆节爱国演讲稿
2014/09/23 职场文书
趣味运动会口号
2015/12/24 职场文书
2019辞职报告范本3篇!
2019/07/23 职场文书
win10音频服务未响应怎么解决?win10音频服务未响应未修复的解决方法
2022/08/14 数码科技