首页 星云 工具 资源 星选 资讯 热门工具
:

PDF转图片 完全免费 小红书视频下载 无水印 抖音视频下载 无水印 数字星空

后端说,单页面SPA和前端路由是怎么回事

编程知识
2024年07月26日 22:02

没有请求的路由

在传统开发中,浏览器点击一个超链接,就会像后端web服务器发送一个html文档请求,然后页面刷新。但开始单页面开发后,就完全不同了。

单页面?这个概念难以理解。我用一个js作为整个web应用,然后再这个js中操作dom变化,以此来实现页面变化。这不叫单页面吗?这叫!但不完善,因为这种方法破坏了浏览器自带的导航功能。比如前进,后退。所以单页面前端应用要解决两件事内容变化导航变化。这是现代前端成立的基础。

想必初次接触vue-routernuxt的人很多对前端路由困惑。明明浏览器地址栏的链接变了,为什么浏览器却没有发送请求出去?至少我是很疑惑的。

这要归功于一个浏览器API,History API。学过wpf的人可能对这玩意不陌生,因为wpf也可以借助导航开发浏览器式应用程序。

导航

导航这种理念包括如下几种功能

  • history.back()
    后退到上一个页面

  • history.forward()
    前进到下一个页面

  • history.go(-2)
    跳转到前两个页面

但下面这三个方法才是实现单页面的关键。因为调用这三个api设置location不会引起浏览器向服务器发送页面请求

  • history.pushState(data, title, url)
  • history.popState()
  • history.replaceState(data, title, url)

history导航是一个栈,这是用来操作栈的方法,入栈、出栈、更新栈顶元素。只不过这个栈里面存放的是页面相关信息。最重要的是pushState,其他的可以暂时不管,也不影响使用。

还有一个关键的,当点击导航浏览器前进和后退按钮,会触发事件window.onpopstate。在这里,我们用js读取导航栈中的信息,并操作dom。这解决了和浏览器导航集成的问题。

这些 API 的主要目的是支持像单页应用这样的网站,它们使用 JavaScript API(如 fetch())来更新页面的新内容,而不是加载整个新页面。

其实到这里,单页面基实现的本原理已经清楚了。

实现一个简单的单页面应用

容器

单页面应用需要一个容器,这里我使用一个div作为页面的容器。

<div></div>

实现页面

页面由模板js代码组成。为了方便书写,各自放在一个script标签中。在模板中声明页面结构,用type="text/html"属性,使得我们可以获得语法感知的提示。然后再定义脚本,其实就是一个函数。在其中读取模板,请求数据,然后渲染页面到容器中

<!-- 模板 -->
	<script type="text/html">
		<h1>首页</h1>
		<h2>这是第一个页面</h2>
		<div></div>
		<button style="background-color: lightcoral;" onclick="route.routeTo('/page2')">跳转到关于页面</button>
	</script>
<!-- 脚本 -->
	<script>
		function loadPage1(){
			//把模板页面替换进容器
			document.getElementById("app").innerHTML=document.getElementById("page1_html").innerHTML;
			//取数据然后生成内容,实际可能有ajax和fetch请求
			var data={
				text:"这是使用js生成的内容",
				id:1
			};
			document.getElementById("content").innerText=JSON.stringify(data);
		}
	</script>

前端路由调度

光有模板和脚本不行,我们还需要一个调度算法,用来调用页面渲染函数、更新导航。就是所谓的前端路由功能了。为了便于观察,我把声明的页面放在一个对象中,这就形成了路由表,方便搜索。当然,不要这个也是可以的,可以手动调用脚本渲染函数loadPage1

const route={
    page:[
        {
            url:"/page1",
            module:loadPage1
        },
        {
            url:"/page2",
            module:loadPage2
        }
    ]
}

有了路由表之后,就可以添加调度算法,根据传入的url,寻找到对应的对象,添加到导航栈中,然后调用脚本渲染函数loadPageXXX

const route={
    routeTo:(url)=>{
        var page=route.page.find(r=>r.url==url);
        if(page==null){
            alert("文件不存在");
        }
        else{
            if(window.location.pathname==url)return;
            history.pushState(page.url,"",page.url);
            page.module("这里可以传参数");
        }
    }
}
//默认跳转到首页
route.routeTo("/page1");

到了这一个,已经实现了单页面。点击跳转按钮,地址栏会变,页面也会变。但有一个问题,点击浏览器导航按钮不管用。这是因为我们还没监听popstate事件处理这个操作。

因为点击导航按钮,地址栏会变,但页面渲染什么内容?这需要我们去处理。所以在这个事件中,我们根据url,从路由表中找到页面,然后渲染出来。

// 处理前进与后退
        window.addEventListener("popstate",(e)=>{
            var page=route.page.find(r=>r.url==e.state);
            if (page) {
                page.module("这里可以传参数");
            }
        })

效果

可以看到,切换页面时,地址栏变了,但并没有网络请求发出

image

完整代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>SPA页面</title>
</head>
<body>
    <div></div>
    <!-- 首页 -->
    <script type="text/html">
        <h1>首页</h1>
        <h2>这是第一个页面</h2>
        <div></div>
        <button style="background-color: lightcoral;" onclick="route.routeTo('/page2')">跳转到关于页面</button>
    </script>
    <script>
        function loadPage1(){
            //把模板页面替换进容器
            document.getElementById("app").innerHTML=document.getElementById("page1_html").innerHTML;
            //取数据然后生成内容,实际可能有ajax和fetch请求
            var data={
                text:"这是使用js生成的内容",
                id:1
            };
            document.getElementById("content").innerText=JSON.stringify(data);
        }
    </script>

    <!-- 关于页 -->
    <script type="text/html">
        <h1>关于</h1>
        <h2>这是第二个页面</h2>
        <div></div>
        <button style="background-color: lightgreen;" onclick="route.routeTo('/page1')">跳转到首页</button>
    </script>
    <script>
        function loadPage2(){
            //把模板页面替换进容器
            document.getElementById("app").innerHTML=document.getElementById("page2_html").innerHTML;
            //取数据然后生成内容,实际可能有ajax和fetch请求
            var data={
                text:"假设这里是网站信息",
                id:2
            };
            document.getElementById("content").innerText=JSON.stringify(data);
        }
    </script>
    
    <!-- 调度 -->
    <script>
        const route={
            page:[
                {
                    url:"/page1",
                    module:loadPage1
                },
                {
                    url:"/page2",
                    module:loadPage2
                }
            ],
            routeTo:(url)=>{
                var page=route.page.find(r=>r.url==url);
                if(page==null){
                    alert("文件不存在");
                }
                else{
                    if(window.location.pathname==url)return;
                    history.pushState(page.url,"",page.url);
                    page.module("这里可以传参数");
                }
            }
        }
        // 处理前进与后退
        window.addEventListener("popstate",(e)=>{
            var page=route.page.find(r=>r.url==e.state);
            if (page) {
                page.module("这里可以传参数");
            }
        })
        //默认跳转到首页
        route.routeTo("/page1");
    </script>
</body>
</html>

结语

从上面的实现中,你们也能看到,单页面应用就是把整个应用程序发送到浏览器,在浏览器里面去运行这个程序。所以相比与一个html文件,一旦应用上规模,体积也相应的会很大。这就牵扯出chunk,把应用分块的概念。第一次请求的时候只把调度部分、首页以及相关几个页面传入到浏览器,后续请求到了没有传输的页面时,再把后续文件传输过来。

由于我们的页面都是在一个网页中,本质上是传输了一个web服务器到浏览器中,在导航时,由js控制渲染。所以过渡效果、过滤器、请求管道、中间件、web服务器具有的东西在网页中实现也就有了可能。

但单页面应用SPA,这个web服务器一切都要归功于浏览器提供的那个关键的功能,History API

From:https://www.cnblogs.com/ggtc/p/18322846
本文地址: http://www.shuzixingkong.net/article/467
0评论
提交 加载更多评论
其他文章 矩阵的奇异值分解(SVD)及其应用
该博客针对矩阵的奇异值分解(SVD)展开介绍,主要介绍了奇异值分解的计算及其几何意义,并基于C++编程语言举例说明了SVD分解的一些应用。
矩阵的奇异值分解(SVD)及其应用 矩阵的奇异值分解(SVD)及其应用 矩阵的奇异值分解(SVD)及其应用
微服务:解决复杂业务的妙方
1 微服务介绍 1)什么是微服务 ​ 微服务(Microservices)是一种软件架构风格,它将一个大型应用程序拆分成许多较小的、松散耦合的、独立运行的服务。这些服务通常围绕特定功能或业务领域组织,可以独立开发、部署、扩展和更新。微服务之间通过轻量级的通信协议(如HTTP/REST、消息队列等)相
微服务:解决复杂业务的妙方 微服务:解决复杂业务的妙方 微服务:解决复杂业务的妙方
.NET 控件转图片
Windows应用开发有很多场景需要动态获取控件显示的图像,即控件转图片,用于其它界面的显示、传输图片数据流、保存为本地图片等用途。 下面分别介绍下一些实现方式以及主要使用场景 RenderTargetBitmap 控件转图片BitmapImage/BitmapSource,在WPF中可以使用Ren
.NET 控件转图片 .NET 控件转图片
【工具】IDEA怎么查看maven依赖链路?
当我在SpringBoot项目中想加个依赖,但是不确定现有依赖的依赖的依赖.....有没有添加过这个依赖,怎么办呢?如果添加过了但是不知道我需要的这个依赖属于哪个依赖的下面,怎么查呢? IDEA中提供了很方便的视图可以满足我们的需求 第一步点击项目右侧的maven 第二步选中Dependencies
【工具】IDEA怎么查看maven依赖链路? 【工具】IDEA怎么查看maven依赖链路? 【工具】IDEA怎么查看maven依赖链路?
Asp .Net Core 系列:详解授权以及实现角色、策略、自定义三种授权和自定义响应
什么是授权(Authorization)? 在 ASP.NET Core 中,授权(Authorization)是控制对应用资源的访问的过程。它决定了哪些用户或用户组可以访问特定的资源或执行特定的操作。授权通常与身份验证(Authentication)一起使用,身份验证是验证用户身份的过程,授权与身
Asp .Net Core 系列:详解授权以及实现角色、策略、自定义三种授权和自定义响应
通过Jupyter Notebook+OpenAI+ollama简单的调用本地模型
通过Jupyter Notebook+OpenAI+ollama简单的调用本地模型 起因是收到了ollama的邮件,貌似支持使用openai来调用本地的ollama下载的模型为自己用 想了下正好试下,因为这几天正好在尝试用Jupyter Notebook来写点调用api的方式来使用大语言模型,看看后
通过Jupyter Notebook+OpenAI+ollama简单的调用本地模型 通过Jupyter Notebook+OpenAI+ollama简单的调用本地模型 通过Jupyter Notebook+OpenAI+ollama简单的调用本地模型
提高 C# 的生产力:C# 13 更新完全指南
前言 预计在 2024 年 11 月,C# 13 将与 .NET 9 一起正式发布。今年的 C# 更新主要集中在 ref struct 上进行了许多改进,并添加了许多有助于进一步提高生产力的便利功能。 本文将介绍预计将在 C# 13 中添加的功能。 注意:目前 C# 13 还未正式发布,因此以下内容
浅谈 I/O 与 I/O 多路复用
1.基础知识 网络编程里常听到阻塞IO、非阻塞IO、同步IO、异步IO等概念,总听别人聊不如自己下来钻研一下。不过,搞清楚这些概念之前,还得先回顾一些基础的概念。 下面说的都是Linux环境下,跟Windows环境不一样哈&#183;☺。 1.1 用户空间和内核空间 现在操作系统都采用虚拟寻址,处理
浅谈 I/O 与 I/O 多路复用 浅谈 I/O 与 I/O 多路复用 浅谈 I/O 与 I/O 多路复用