首发于:http://www.blazor.group:8000/topic/reply?tpid=9

开门见山,不介绍,不废话

建议食用本文前先食用 https://www.cnblogs.com/wzxinchen/p/12082136.html

正常情况下,Blazor 的界面是怎样刷新的?

Blazor 绑定(绑定就是刷新)机制有以下几种

  1. 首次加载时的自动绑定
  2. 调用 StateHasChanged 强制重新绑定(重新绑定即刷新)
  3. 注册事件自动刷新

对于第三点,注册事件是指类似于以下代码

<BMenu OnClick="ShowMenu"></BMenu>

注意这代码是瞎写的,仅仅为了示例

在上面代码中,注册了 OnClick 事件,处理程序为 ShowMenu

OnClick 既然是事件,那它总得有个方法签名,来规定 ShowMenu 的方法签名是什么

一般来说,有两种方法签名:

  1. 直接就是一个委托
  2. EventCallBack结构体

对于第一种情况,OnClick 事件的定义往往是

[Parameter]
public Action OnClick {get;set;}

对于第二种情况,OnClick 事件的定义往往是

[Parameter]
public EventCallBack OnClick {get;set;}

这两种情况在调用方注册的方法是一样的,那么这两种情况有什么不一样的?

不同点很小,第一种情况没有什么好说的,第二种情况,EventCallBack 内部调用了 StateHasChanged。也就是说,如果事件定义是第二种,那么在方法执行完后会自动刷新一次,如果是第一种,则需要手动刷新

默认情况下,Blazor 的刷新机制有怎样的问题?

  1. 通过第一段我们知道,如果我们注册了一个事件,而且这个事件是 EventCallBack,那么在方法调用完成之后会自动刷新

    如果我们注册的不是 OnClick 事件,而是 OnMouseMove 事件呢?那么 StateHasChanged 事件会被不停得调用,会给服务端造成极大压力
  2. 我们在编写页面的时候,往往会在不同的地点手动调用 StateHasChanged 方法来刷新界面,假设 A 方法调用了 B 方法,B 又调用 C,并且 A 是类型为 EventCallBack 事件处理程序,然后 B、C里面最开始都调用了 StateHasChanged,这会刷新多少次?你的服务器资源会被吃光。

Blazor 如何解决这个问题?

Blazor 的 ComponentBase 类中,提供了 ShouldRender 这个方法,当这个方法返回为 false 时,不会执行渲染,即使你调用了 StateHasChanged,仍然不会渲染,这相当于让你来决定,什么时候才是真的需要渲染,什么时候调用 StateHasChanged 才会生效。

你的 Blazui 组件界面为什么没刷新?

基于 Blazor 的解决办法,当组件第一次渲染完成之后,ShouldRender 会返回为 false,然后后面调用都返回 false,对于任意一个组件,若出现没刷新的情况下,请考虑这个因素。

那么,如何让 Blazui 组件进行刷新呢?方法很简单,调用该组件的 MarkAsRequireRender 方法,标记该组件需要刷新。注意这个方法只是标记为需要刷新,如果不是自动调用的 StateHasChanged 方法,那么你需要手动调一次才刷新,这个方法 Blazui 封装为 Refresh 方法,内部直接调的 StateHasChanged 方法

我调用了 MarkAsRequireRender 方法,为什么仍然没刷新?

到了这步,仍然没解决,可能会有些头疼

因为要刷新的组件的父组件没刷新

这里涉及到一个很重要的概念,Blazor 中,一个页面中的所有组件,是一颗树

可以简单理解为二叉树或是N叉树,每一个节点,就是一个组件,这个节点下的直属节点,是这个组件的子组件

<A>
<B>
<C>
</C>
</B>
</A>

A 是 B 的父组件,B 是 C 的父组件,要刷 C,你必须刷 B,注意这里的刷 B 并不会刷新 B 本身,而是刷的 B 的直属子组件,现在,你应该可以理解,我要刷新 B,我又该刷谁?当然是刷 B 的父组件

为什么会这样?因为你如果刷新 B,那么 B 所需要的参数根本不会更新,这样一来也根本无法刷新数据,它的主要目的,是更新 B 的直属子组件的所有参数,这样才能刷新 B 的直属子组件

因为当前页面所属组件没刷新

Blazor 中,一切皆组件,当前页面它仍然是个组件,按照第一点讲到的,你大概可以理解这种情况该怎么办,很简单,设置当前页面为需要刷新即可

若当前页面没有继承 BComponentBase 这个类,则不需要考虑这个情况。

你的异步调用先后顺序错误

这种情况是比较坑的,不容易发现。

考虑下面的代码

public void Test(){
Task.Factory.StartNew(()=>{
//从数据库拉数据
}); }

Test 是 OnClick 事件的处理方法,很简单的一个情况,假设它不存在我们之前说的所有情况,这样写,仍然有概率不刷新,注意,是有概率,只是这概率特高。

当这个方法执行完并且开始执行 StateHasChanged 时,异步任务里面的拉取数据的代码执行完了吗?不一定,天知道。

通常情况是没执行完,那么就是说,当刷新时,数据压根就没拉到,这样的话,你的界面当然不会更新。

public async Task Test(){
await Task.Factory.StartNew(()=>{
//从数据库拉数据
}); }

改成这样即可

Blazui 常见问题:我更新了数据,为什么界面没刷新?的更多相关文章

  1. Working with Data » Getting started with ASP.NET Core and Entity Framework Core using Visual Studio » 更新关系数据

    Updating related data¶ 7 of 7 people found this helpful The Contoso University sample web applicatio ...

  2. 使用TSQL查询和更新 JSON 数据

    JSON是一个非常流行的,用于数据交换的文本数据(textual data)格式,主要用于Web和移动应用程序中.JSON 使用“键/值对”(Key:Value pair)存储数据,能够表示嵌套键值对 ...

  3. ReactNative新手学习之路06滚动更新ListView数据的小示例

    本节带领大家学习使用ListView 做一个常用的滚动更新数据示例: 知识点: initialListSize={200} 第一次加载多少数据行 onEndReached={this.onEndRea ...

  4. [渣译文] 使用 MVC 5 的 EF6 Code First 入门 系列:为ASP.NET MVC应用程序更新相关数据

    这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译,这里是第八篇:为ASP.NET MVC应用程序 ...

  5. Contoso 大学 - 6 – 更新关联数据

    原文 Contoso 大学 - 6 – 更新关联数据 By Tom Dykstra, Tom Dykstra is a Senior Programming Writer on Microsoft's ...

  6. SQL语法集锦一:显示每个类别最新更新的数据

    本文转载http://www.cnblogs.com/lxblog/archive/2012/09/28/2707504.html (1)显示每个类别最新更新的数据 在项目中经常遇到求每个类别最新显示 ...

  7. highcharts图表组件入门教程:如何监听柱状图柱子点击事件动态更新当前数据点数值和所对应X轴刻度

    highcharts图表组件入门教程:如何监听柱状图柱子点击事件动态更新当前数据点数值和所对应X轴刻度 作者:highcharts | 时间:2014-6-11 14:07:05 | [小  大] | ...

  8. ASP.NET MVC应用程序更新相关数据

    为ASP.NET MVC应用程序更新相关数据 这是微软官方教程Getting Started with Entity Framework 6 Code First using MVC 5 系列的翻译, ...

  9. C++彩色数据流动界面

    一个数据流动界面 #include <windows.h> #include <time.h> #include <cstdio> #include <str ...

随机推荐

  1. spring Cloud-eureka的保护模式

    eureka的首页出现以下警告 EMERGENCY! EUREKA MAY BE INCORRECTLY CLAIMING INSTANCES ARE UP WHEN THEY'RE NOT. REN ...

  2. centos7搭建hadoop2.10高可用(HA)

    本篇介绍在centos7中搭建hadoop2.10高可用集群,首先准备6台机器:2台nn(namenode);4台dn(datanode):3台jns(journalnodes) IP hostnam ...

  3. tcpdump用法说明

    tcpdump采用命令行方式对接口的数据包进行筛选抓取,其丰富特性表现在灵活的表达式上. 不带任何选项的tcpdump,默认会抓取第一个网络接口,且只有将tcpdump进程终止才会停止抓包. 例如: ...

  4. xhsell关闭jupyter仍然运行的命令

    nohup jupyter notebook & nohup 和 &都是linux的命令 1.& 当在前台运行某个作业时,终端被该作业占据:可以在命令后面加上& 实现后 ...

  5. python sqlalchemy mysql 自动映射

    SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作 简言之便是:将对象转换成SQL,然后使用数据API执行SQL并获取执行结果 ...

  6. Kubernetes-Ingress资源详解

    什么是Ingress #阿里云称之为ingress路由!在 Kubernetes 集群中,主要用于接入外部请求到k8s内部,Ingress是授权入站连接到达集群服务的规则集合,为您提供七层负载均衡能力 ...

  7. QIIME2使用方法

    激活qiime2的执行环境:source activate qiime2-2019.4如何查看conda已有的环境:conda info -e 以下分析流程参考:https://docs.qiime2 ...

  8. TypeScript躬行记(6)——高级类型

    本节将对TypeScript中类型的高级特性做详细讲解,包括交叉类型.类型别名.类型保护等. 一.交叉类型 交叉类型(Intersection Type)是将多个类型通过“&”符号合并成一个新 ...

  9. ReLU函数

    Rectifier(neural networks) 在人工神经网络中,rectfier(整流器,校正器)是一个激活函数,它的定义是:参数中为正的部分. , 其中,x是神经元的输入.这也被称为ramp ...

  10. HDU 6602 Longest Subarray (线段树)

    题意: 1e5的数组,c(1e5)种数字求最长的子串,使得其中每个出现的数字出现的次数为0次或者大于k次 思路: 枚举右端点i,维护当前右端点时,每个左端点的可行元素数量,当且仅当可行元素为c时更新答 ...