Blazor 机制初探以及什么是前后端分离,还不赶紧上车?
标签: Blazor .Net
上一篇文章发了一个 BlazAdmin 的尝鲜版,这一次主要聊聊 Blazor 是如何做到用 C# 来写前端的,传送门:https://www.cnblogs.com/wzxinchen/p/12057171.html
飚车前
需要说明的一点是,因为我深入接触 Blazor 的时间也不是多长,顶多也就半年,所以这篇文章的内容我不能保证 100% 正确,但可以保证大致原理正确
另外,具有以下条件的园友食用这篇文章会更舒服:
- 了解 Http 请求响应模型及 Http 协议
- 有足够的微软技术栈 Web 开发经验,例如 MVC、WebApi 等
- 有按照微软的 Blazor 官方文档进行入门的实战操作,传送门:https://docs.microsoft.com/zh-cn/aspnet/core/blazor/get-started?view=aspnetcore-3.1&tabs=visual-studio
- 有自己研究过 Blazor 生成的代码
- 有过 SignalR 或 WebSocket 使用经验
建议结合 AspNetCore 源码看这篇文章,我不能贴出所有源码,源码需要编译过才能看,不然会很麻烦,但编译这事比较难,编译源码比看源码难多了,这儿是一位园友的源码编译教程:https://www.cnblogs.com/ZaraNet/p/12001261.html
天底下没有新鲜事儿,Blazor 看着神奇,其实也没啥黑科技,它跑不掉 Http 协议,也跑不掉 Html
开始发车
Blazor 服务端渲染过程
当您打开一个服务端渲染的 Blazor 应用时:
浏览器 -->> 服务器: 建立 WebSocket 连接
服务器 -->> 浏览器: 发送首页 HTML 代码
loop 连接未断开
Note left of 浏览器: 浏览器JS捕获用户输入事件
浏览器 -->> 服务器: 通知服务器发生了该事件
Note right of 服务器: 服务器 .Net 处理事件
服务器-->>浏览器: 发送有变动的 HTML 代码
Note left of 浏览器: 浏览器JS渲染变动的 HTML 代码
end
有以下几点需要注意:
- WebSocket 连接采用 SignalR 来建立,如果浏览器不支持 WebSocket,SignalR 会采用其他技术建立
- 浏览器捕获用户输入是使用 Javascript进行捕获的
- 服务器处理客户端事件完成后,会生成新的 HTML 结构,然后将这个结构与老的结构进行对比,得到有变动的 HTML 代码
- Blazor 服务端渲染版采用在服务器端维护一个虚拟 DOM 树来实现上述操作
- “通知服务器发生了该事件”这一步里,从原理上来说类似于 WebForm 的 PostBack 机制,不同点在于,Blazor 只告诉服务器是哪个 DOM 节点发生了什么事件,这个传输量是极小的。
服务端渲染的基本原理就是这样,下面我们详细讨论
Blazor 路由渲染过程
当我们通过 NavigationManager 去改变路由地址时,大概流程如下
st=>start: 服务器启动
rt=>operation: 初始化 Router 组件,Router 内部注册 LocationChanged 事件
op1=>operation: LocationChanged 事件中根据路由查找对应的组件,默认触发首页组件
queue=>operation: 加入渲染队列
render=>operation: 一直进行渲染及比对,直到队列中所有的组件全部渲染完
diff=>operation: 将比对的差异结果更新至浏览器
e=>end: 等待下一次路由改变,继续触发 LocationChanged 事件
st->rt->op1->queue->render->diff->e
这里的 Router 组件,就是我们经常用到的,看看下面的代码,是不是很熟悉?
<Router AppAssembly="@typeof(Program).Assembly">
<Found Context="routeData">
<RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
</Found>
<NotFound>
<LayoutView Layout="@typeof(MainLayout)">
<p>Sorry, there's nothing at this address.</p>
</LayoutView>
</NotFound>
</Router>
Router 组件部分代码
public class Router : IComponent, IHandleAfterRender, IDisposable
{
public void Attach(RenderHandle renderHandle)
{
_logger = LoggerFactory.CreateLogger<Router>();
_renderHandle = renderHandle;
_baseUri = NavigationManager.BaseUri;
_locationAbsolute = NavigationManager.Uri;
//注册 LocationChanged 事件
NavigationManager.LocationChanged += OnLocationChanged;
}
private void OnLocationChanged(object sender, LocationChangedEventArgs args)
{
_locationAbsolute = args.Location;
if (_renderHandle.IsInitialized && Routes != null)
{
Refresh(args.IsNavigationIntercepted);
}
}
private void Refresh(bool isNavigationIntercepted)
{
var locationPath = NavigationManager.ToBaseRelativePath(_locationAbsolute);
locationPath = StringUntilAny(locationPath, _queryOrHashStartChar);
var context = new RouteContext(locationPath);
Routes.Route(context);
..........
var routeData = new RouteData(
context.Handler,
context.Parameters ?? _emptyParametersDictionary);
//此处开始渲染,Found 是一个 RenderFragment<RouteData> 委托,是我们在调用的时候指定的那个
_renderHandle.Render(Found(routeData));
..........
}
}
Blazor 组件渲染过程
要开始飚车了,握紧方向盘,不要翻车。
这部分可能会比较难,如果你发现你看不懂的话就先尝试自己写个组件玩玩。
在 Blazor 中,几乎一切皆组件。首先我们得提到一个 Blazor 组件的几个关键方法,部分方法也是它的生命周期
- OnInitialized、OnInitializedAsync:仅在第一次实例化组件时,才会调用这些方法一次。注意,该方法调用时参数已经设置,但没有渲染。
- SetParametersAsync:该方法可以让您在设置参数之前做一些事
- OnParametersSetAsync、OnParametersSet:每一次参数设置完成之后都会调用
- OnAfterRender、OnAfterRenderAsync:在组件渲染完成之后触发
- ShouldRender:如果该方法返回 false,则组件在第一次渲染完成后不会执行二次渲染
- StateHasChanged:强制渲染当前组件,如果 ShouldRender 返回的是 false,则不会强制渲染
- BuildRenderTree: 该方法一般情况下我们用不到,它的作用是拼接 HTML 代码,由 VS 自动生成的代码去调用它
另有一个关键的结构体 EventCallBack
,还有一个关键的委托RenderFragment
,它俩非常重要,前者可能见得比较少,后者基本上玩过 Blazor 的园友都知道。
上面提到的关键点,有个印象即可,下面将开始飚车,我们将重点讨论那个流程图中渲染对比的那部分,但将忽略浏览器捕获事件这一步,我不能贴太多的源码,尽可能用流程图表示
主要生命周期过程
st=>start: 开始渲染
isfirst=>condition: 是否首次渲染
init=>operation: 调用 OnInitialized 方法
initAsync=>operation: 调用 OnInitializedAsync 方法
onSetParameter=>operation: 调用 OnParametersSet 方法
setParameter=>operation: 调用 SetParametersAsync 方法
stateHasChanged=>operation: 调用 StateHasChanged 方法
st->setParameter->isfirst->init->initAsync->onSetParameter
onSetParameter->stateHasChanged
isfirst(yes)->init
isfirst(no)->onSetParameter
需要注意的是这个流程中没有 OnAfterRender
方法的调用,这个将在下面讨论
StateHasChanged 方法
这个方法至关重要,就比如上图中最终只到了 StateHasChanged
方法,就没了下文,我们来看看这个方法里面有什么
st=>start: 开始
isfirst=>condition: 是否首次渲染
should=>condition: ShouldRender 为True?
queue=>operation: 进入渲染队列
render=>operation: 开始循环渲染队列的数据
after=>operation: 触发 OnAfterRender 方法
e=>end: 结束
st->isfirst
queue->render->after->e
isfirst(yes)->queue
isfirst(no)->should
should(yes)->queue
should(no)->e
至此,我们基本把一个组件的生命周期的那几个方法讨论完了,除了一些异步版本的,逻辑都差不多,没有写进来
渲染队列时都干了啥?
嗯对,这是重点
st=>start: 开始渲染队列
queue=>condition: 队列还有组件?
read=>operation: 从队列获取组件
swap=>operation: 备份当前 DOM 树及清空
render=>operation: 调用组件的 RenderFragment 委托获取新的 DOM 树
diff=>operation: 与备份的树对比
append=>operation: 将对比结果存入列表
display=>operation: 将列表中的所有对比结果发送至浏览器
e=>end: 结束
st->queue
read->swap->render->diff->append->queue
queue(yes)->read
queue(no)->display->e
为了图好看点(好吧现在其实也不好看),我把流程缩短了一点,有以下几点需要注意:
- 渲染开始之前是将当前树赋值成了旧的树,然后再将当前树清空
- 组件的
RenderFragment
委托在大多数情况下就是组件的ChildContent
属性的值,玩过的都知道几乎每个组件都有自己的ChildContent
。- 同时
RenderFragment
也有可能是ComponentBase
类中的一个私有属性,详见下面的代码。当然也有可能是其他的,限于篇幅,不细说RenderFragment
委托输入的参数就是当前这颗树- 如果您在组件中调用了子组件,并且这个子组件还有自己的内容,那么 VS 会生成调用这个组件的代码,并且为这个组件添加
ChildContent
属性,内容就是子组件自己的内容,详见代码
下面是 ComponentBase
的部分代码,上文提到的私有属性就是 _renderFragment
,这个私有属性仅在此处被赋值,可以看到这个属性内部调用了 BuildRenderTree
方法
public abstract class ComponentBase : IComponent, IHandleEvent, IHandleAfterRender
{
private readonly RenderFragment _renderFragment;
/// <summary>
/// Constructs an instance of <see cref="ComponentBase"/>.
/// </summary>
public ComponentBase()
{
_renderFragment = builder =>
{
_hasPendingQueuedRender = false;
_hasNeverRendered = false;
BuildRenderTree(builder);
};
}
}
针对最后一点,举个例子
下面是 NavMenu.razor
组件的 Razor 代码
<BMenu>
<BMenuItem Route="button">Button 按钮</BMenuItem>
</BMenu>
下面是 VS 生成的代码
public partial class NavMenu : Microsoft.AspNetCore.Components.ComponentBase
{
protected override void BuildRenderTree(Microsoft.AspNetCore.Components.Rendering.RenderTreeBuilder __builder)
{
__builder.OpenComponent<BMenu>(1);
__builder.AddAttribute(4, "ChildContent", (Microsoft.AspNetCore.Components.RenderFragment)((__builder2) => {
__builder2.OpenComponent<BMenuItem>(6);
__builder2.AddAttribute(7, "Route", "button");
__builder2.AddAttribute(8, "ChildContent", (Microsoft.AspNetCore.Components.RenderFragment)((__builder3) => {
__builder3.AddMarkupContent(9, "Button 按钮");
}
));
__builder2.CloseComponent();
}
}
}
可以看到,NavMenu.razor
使用了 BMenu
这个组件,BMenu
又使用了 BMenuItem
这个组件,共套了两层,因此生成了两个 ChildContent
的属性,而且属性类型都是 Microsoft.AspNetCore.Components.RenderFragment
到这儿为止,Blazor 的大概机制基本讨论了一半,接下来讨论上个流程图中的对比那一步,看看 Blazor 是如何进行的对比
这里不细说,因为确实太复杂我也没搞清楚,只说个大概流程,需要说明的一点是 Blazor 的对比是基于序列号的,序列号是什么?大家一定注意到上面代码中的 __builder.AddAttribute(4
中的这个 4 了,这个 4 就是序列号,然后每个序列号对应的内容称为帧,简而言之是通过判断每个序列号对应的帧是否一致来对比是否有改动
st=>start: 开始对比
seq=>operation: 循环每帧
compare=>condition: 序列号是否一致?
isComponent=>condition: 该帧是否都为组件?
render=>operation: 渲染该组件
compareParameter=>condition: 两边组件的参数是否有变化?
skip=>operation: 跳过该帧
setParameter=>operation: 设置新组件的参数,进入该组件的生命周期流程
currentSkip=>operation: 机制过于复杂,不讨论
e=>end: 对比结束
endSeq=>operation: 结束循环
st->seq->compare
compare(yes)->isComponent
compare(no)->currentSkip
isComponent(yes)->render->compareParameter
isComponent(no)->currentSkip
compareParameter(yes)->setParameter->endSeq->e
compareParameter(no)->skip
流程图总算画完了,大概有以下几点需要注意:
- 实际的对比过程是很复杂的,流程图是简化了再简化的结果,这篇文章的几个流程图需要结合在一起理解才行
- 当走到设置新组件的参数这一步时,继续往下其实就是进入了新组件的生命周期流程,这个流程跟上面的生命周期流程是一样的
- 结合所有流程图来看,如果只是组件本身重新渲染,那么组件本身设置参数的方法不会被触发,必须是它的父组件被渲染,才会触发它自己的设置参数的方法
- 对比组件参数这一步,流程图比较笼统。我们可以简单的认为,没有组件的参数是不变化的,它的对比流程过于细节,我觉得没必要写进来。
渲染到此结束,下面就来谈谈 Blazor 会让我们遇到的问题
Blazor 的不足
优势我们就不谈了,我们来谈谈一个比较隐藏但又不容易解决的不足,这个不足就是我们一不小心就让我们的 Blazor 应用变得卡,而且还比较不容易解决,这个问题在服务端渲染的应用中尤其严重。
结合第一张流程图,浏览器产生任何事件都会发送到服务器端,想象一下你注册了一个 onmousemove
事件的话,还要不要活了?所以,大规模触发的事件尽量少注册,这里面的网络传输成本是很大的,而且也会给你的服务端造成很大的压力。
Blazor 应用变卡一般有以下几种情况,我们只讨论服务端应用的情况
- 服务器端已经挂了,这种情况其实浏览器端会完全失去响应,除非你刷新
- 你的代码有问题或你引用的库的代码有问题,导致进入死循环或循环次数非常多
第一点无所谓,第二点是要命的,至少对于我来说,一旦 Blazui 或 BlazAdmin 出现了卡的情况,会非常头疼,但实际上大多数情况都是第二种中,原因在于:
结合所有流程图来看,Blazor 完成渲染才会发送至浏览器,那么完成渲染的标准就是渲染队列被清空,那如果一直无法清空呢?体现出来就是死循环,或者说发生了一次点击事件结果循环了十次,这明显不科学(你故意的例外),而渲染队列被加入新东西大多数情况下是因为调用了 StateHasChanged
并且 ShuoldRender
返回了 true
,或者是因为使用了 EventCallBack
,这些代码所在的地方你全都难以调试
因为这些代码不是你的代码,所以你的断点也没处打,目前的 Blazor 不会告诉你到底是哪个组件哪行代码引起的死循环
还欠了点东西
还有一个关键的东西是 EventCallBack
,一次写太多了,不想写了
园友如果有兴趣的话可以继续把这个写了
有任何问题可进QQ群交流:74522853
什么是前后端分离?
Blazor 出来的时候一堆人说什么 WebForm 又来了,Silverlight 又来了,还有啥啥乱七八糟的,最让我不能理解的是另一种说法:
前后端分离搞得好好的,微软为什么又要把前后端合在一起?
我不敢瞎说,我找了一篇文章:https://www.jianshu.com/p/bf3fa3ba2a8f
下面是摘抄的内容
1.首先要知道所有的程序都是一数据为基础的,没有数据的程序没有实际意义,程序的本质就是对程序的增删改查。
2.前后端分离就是把数据操作和显示分离出来。前端专注做数据显示,通过文字,图片或者图标等方式让数据形象直观的显示出来。后端专注做数据的操作。前端把数据发给后端,有后端对数据进行修改。
3.后端一般用java,c#等语言,现在的node属于JavaScript也能进行后端操作,此处不意义裂解语言。后端来进行数据库的链接,并对数据进行操作。
4.后端提供接口给前端调用,来触发后端对数据的操作。
基本原理就是这样,可能语言上不准确,思想是没有问题的。
作者:前端developer 链接:https://www.jianshu.com/p/bf3fa3ba2a8f 来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
重点在于第二点,前后端分离就是把数据操作和显示分离出来,Blazor 并没有有非要让你用 .Net 写后端
第三点也说了,前端一般是 JS,那现在把 JS 换成 .Net 并没有什么不一样
Blazor 机制初探以及什么是前后端分离,还不赶紧上车?的更多相关文章
- 谷粒商城为什么要选择后天管理用vue前后端分离,而商城页面选择Thymeleaf类?
1.最初使用jsp开发web项目 还记得刚学习Java的时候,自己开发一个web项目,那时候前后端分离还不流行,为了在页面中显示动态数据,最终的形式也就是以动态网页响应用户,Java一种技术jsp,J ...
- 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之十二 || 三种跨域方式比较,DTOs(数据传输对象)初探
更新反馈 1.博友@落幕残情童鞋说到了,Nginx反向代理实现跨域,因为我目前还没有使用到,给忽略了,这次记录下,为下次补充.此坑已填 2.提示:跨域的姊妹篇——<三十三║ ⅖ 种方法实现完美跨 ...
- 从壹开始前后端分离 [ vue + .netcore 补充教程 ] 二七║ Nuxt 基础:框架初探
缘起 哈喽大家好,又是周四了,俗话说周四来了,周末还远么哈哈,老张我也想下周请假,来个16天的大长假哟,不知道大家是怎么请假的,近来发现文章下边已经没有人评论了,赶脚比较凄凉了,大家看到的麻烦点个赞呀 ...
- 前端和后端采用接口访问时的调用验证机制(基于JWT的前后端验证)(思路探讨)
说明:基于前后端,尤其是使用Ajax请求的接口,现在市面上网页上调用的Ajax基本都是没有验证的,如果单独提取之后可以无线的刷数据. 继上一篇http://www.cnblogs.com/EasonJ ...
- 前后端分离中的无痛刷新token机制
今天我们来说一说前后端分离中的无痛刷新token机制 博主先来分享一波福利,最近挖到的宝藏,刚开始学Java的同学看 https://www.bilibili.com/video/BV1Rx41187 ...
- 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 二十五║初探SSR服务端渲染(个人博客二)
缘起 时间真快,现在已经是这个系列教程的下半部 Vue 第 12 篇了,昨天我也简单思考了下,可能明天再来一篇,Vue 就基本告一段落了,因为什么呢,这里给大家说个题外话,当时写博文的时候,只是想给大 ...
- Session与Token认证机制 前后端分离下如何登录
字号 1 Web登录涉及到知识点 1.1 HTTP无状态性 HTTP是无状态的,一次请求结束,连接断开,下次服务器再收到请求,它就不知道这个请求是哪个用户发过来的.当然它知道是哪个客户端地址发过来的 ...
- 从壹开始前后端分离 [ Vue2.0+.NET Core2.1] 十九║Vue基础: 样式动态绑定+生命周期
回顾 哈喽大家好,前后端分离系列文章又开始了,今天周一,还是感谢大家花时间来观看我写的博客,周末呢,没有写文章,但是也没有闲着,主要是研究了下遗留问题,看过之前文章的应该知道,之前的在AOP使用Red ...
- 【转】从MVC到前后端分离
1. 理解MVC MVC是一种经典的设计模式,全名为Model-View-Controller,即模型-视图-控制器. 其中,模型是用于封装数据的载体,例如,在Java中一般通过一个简单的POJO(P ...
随机推荐
- 【转载】TCP协议
首部格式 图释: 各个段位说明: 源端口和目的端口:各占 2 字节.端口是传输层与应用层的服务接口.传输层的复用和分用功能都要通过端口才能实现 序号:占 4 字节.TCP 连接中传送的数据流中的每一个 ...
- Java中String对象的存储位置(学习笔记)
首先,String是final修饰的.immutable对象,它以一个个字符的方式存储在字符数组中.其次,String类型创建对象有两种方式:①通过字面量赋值:会先去常量池中查找是否存在相同的字符串, ...
- Gamification vs. Game-Based Learning
http://www.immersedgames.com/gamification-vs-game-based-learning/ With the growth in popularity of v ...
- [译] 在 UNIX 中,一切皆文件
原文地址:In UNIX Everything is a File 原文作者:ph7spot.com 译文出自:掘金翻译计划 本文永久链接:github.com/xitu/gold-m… 译者:pmw ...
- POJ3259-Wormholes-( spfa || Bellman_Ford )
题意:有n块田,之间有m条无向边表示路径,权值表示走过需要花费的时间.有w对虫洞,虫洞是单向的,表示穿越一定时间到过去,并且回到虫洞指向的点,问一个人有没有可能通过虫洞回到某个起点,并且在从这个起点出 ...
- 享元模式(Flyweight Pattern)
定义: 采用一个共享来避免大量拥有相同内容对象的开销.这种开销中最常见.直观的就是内存的损耗.享元模式以共享的方式高效的支持大量的细粒度对象. 享元的英文是flyweight,是一个来自体育方面的专业 ...
- [Beta]Scrum Meeting#4
github 本次会议项目由PM召开,时间为5月9日晚上10点30分 时长15分钟 任务表格 人员 昨日工作 下一步工作 木鬼 撰写博客整理文档 撰写博客整理文档 swoip 改进界面 改进界面 bh ...
- 剑指offer:把数组排成最小的数
题目描述: 输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个.例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323. 思路分析: ...
- java和vue2.0
1 java中的el表达式${对象.属性}和vue中的双向数据绑定{{mode.xx}}感觉有点类似 2 java中 request.setAttribute("hots", l ...
- “sockaddr”: “struct”类型重定义的错误的解决办法《转》
原帖地址:https://blog.csdn.net/clever101/article/details/100163301 windows.h和winsock2.h存在有类型重定义,往往体现在VC程 ...