一个按钮,点击执行一个任务。我们可能直接在它的 Click 事件中写下了执行任务的代码。

一般我们无需担心这样的代码会出现什么问题——但是,这样的好事情只对同步任务有效;一旦进入了异步世界,这便是无尽的 BUG!


 

重新进入(Reentrancy)

private void Button_Click(object sender, RoutedEventArgs e)
{
DoSomething();
} private void DoSomething()
{
// 同步任务。
}

▲ 以上,在按钮点击事件中执行同步任务

上面的代码,无论我们在界面上多么疯狂地点击按钮,因为 UI 会在任务执行的过程中停止响应,所以 DoSomething 只会依次执行(还会偶尔忽略一些)。这通常不会造成什么问题,但如果 DoSomething 变成异步的 DoSomethingAsync(就像下面那样),那么情况就变得不同了。

private async void Button_Click(object sender, RoutedEventArgs e)
{
await DoSomethingAsync();
} private async Task DoSomethingAsync()
{
// 异步任务。
}

▲ 以上,在按钮点击事件中执行异步任务

由于任务执行的过程中 UI 依然是响应的,DoSomethingAsync 会因此在每一次点击的时候都进入。在异步任务结束之前重新进入此异步任务的过程,叫做重新进入(Reentrancy)。

重新进入的五种方式

微软在 Handling Reentrancy in Async Apps (C#) 一文中给出了重新进入的三种方式:

  1. 禁用“开始”按钮
  2. 取消和重启操作
  3. 运行多个操作并将输出排入队列

从语言描述中就能知道除了第 2 点看起来具有通用性外,其他两点只为了解决文章中面临的“输出网页列表”问题。第 1 点其思想可以重用,但第 3 点就很难抽取公共的重新进入思想。于是,我总结其前两点,再额外补充两种重新进入的方式,和不处理一起作为五种不同的处理方法。

  • 禁用重新进入
  • 并发
  • 取消然后重启操作
  • 将异步任务放入队列中依次执行
  • 仅执行第一次和最后一次

禁用重新进入

禁用是最直接最简单也最彻底的重新进入问题解决办法。

Button.IsEnabled = false;
await DoSomethingAsync();
Button.IsEnabled = true;

既然重新进入可能出问题,那我们就禁止重新进入好了……

并发

当然,不处理也是一种方法。这意味着我们需要真的考虑 DoSomethingAsync 并发造成的影响。

取消然后重启操作

取消,然后重新执行一次,这也是常见的重新进入类型。浏览器或者资讯类 APP 中的刷新功能就是这种重新进入方式最常见的应用场景,用户重新执行一次刷新,可能因为前面那一次(因为网络问题或其他原因)太慢,所以重新开始。

将异步任务放入队列中依次执行

放入队列中是因为此异步任务的顺序是很重要的,要求每一次执行且保持顺序一致。典型的应用场景是每一次执行都需要获取或生成一组数据输出(到屏幕、文件或者其他地方)。

仅执行第一次和最后一次

如果用户每一次执行此异步任务都会获取当前应用程序的最新状态,然后根据最新状态执行;那么如果状态更新了,对旧状态执行多少次都是浪费的。

比如保存文件的操作。第一次进入异步任务的时候会进行保存,如果保存过程没有结束又触发新的保存,则等上一次保存结束之后再执行保存操作即可。而如果第一次保存没有结束的时候又触发非常多次的保存,也只需要在第一次结束之后再保存一次即可,毕竟既然最后一次保存时的状态已经是最新状态,不需要再把之前旧的状态保存一次。


参考资料

异步任务中的重新进入(Reentrancy)的更多相关文章

  1. 你不知道的this—JS异步编程中的this

    Javascript小学生都知道了javascript中的函数调用时会 隐性的接收两个附加的参数:this和arguments.参数this在javascript编程中占据中非常重要的地位,它的值取决 ...

  2. Android异步回调中的UI同步性问题

    Android程序编码过程中,回调无处不在.从最常见的Activity生命周期回调开始,到BroadcastReceiver.Service以及Sqlite等.Activity.BroadcastRe ...

  3. [Effective JavaScript 笔记]第62条:在异步序列中使用嵌套或命名的回调函数

    异步程序的操作顺序 61条讲述了异步API如何执行潜在的代价高昂的I/O操作,而不阻塞应用程序继续处理其他输入.理解异步程序的操作顺序刚开始有点混乱.例如,下面的代码会在打印"finishe ...

  4. [置顶] Ajax程序:处理异步调用中的异常(使用Asp.Net Ajax内建的异常处理方法)

    无论在Window应用程序,还是Web应用程序以对用户友好的方式显示运行时的异常都是很有必要,尤其对于可能有很多不确定因素导致异常的Web应用程序;在传统的Web开发中,处理异常的方式——设计专门一个 ...

  5. 使用domain模块捕获异步回调中的异常

    和其他服务器端语言相比,貌似node.js 对于异常捕捉确实非常困难. 首先你会想到try/catch ,但是在使用过程中我们会发现并没有真正将错误控制在try/catch 语句中. 为什么? 答案是 ...

  6. 异步请求中jetty处理ServletRequestListener的坑

    标题起得比较诡异,其实并不是坑,而是jetty似乎压根就没做对异步request的ServletRequestListener的特殊处理,如果文中有错误欢迎提出,可能自己有所疏漏了. 之前遇到了一个b ...

  7. Effective JavaScript Item 63 注意异步调用中可能会被忽略的异常

    异常处理是异步编程的一个难点. 在同步的代码中,异常可以非常easy地通过try catch语句来完毕: try { f(); g(); h(); } catch (e) { // handle an ...

  8. 【转】C# Async/Await 异步编程中的最佳做法

    Async/Await 异步编程中的最佳做法 Stephen Cleary 近日来,涌现了许多关于 Microsoft .NET Framework 4.5 中新增了对 async 和 await 支 ...

  9. [知识库:python-tornado]异步调用中的上下文控制Tornado stack context

    异步调用中的上下文控制Tornado stack context https://www.zouyesheng.com/context-in-async-env.html 这篇文章真心不错, 非常透彻 ...

随机推荐

  1. Lucene 初步 之 HelloWorld

    万恶的源头 HelloWorld 要完成lucene 的配置 需要几个jar包 (如果需要可以留言我私发) 创建索引API分析: 1. Directory: 类代表一个Lucene索引的位置,FSDi ...

  2. sql数据类型转换函数

    1.CAST()CAST (<expression> AS <data_ type>[ length ]) 2.CONVERT()CONVERT (<data_ type ...

  3. Selenium元素定位问题

    定位元素时,遇到一些诡异事件: 明明就是通过ID定位的,但是就是没有定位到该元素呢? 1.通过element.find_elements_by_xxx()获取该元素的个数,试试是否有获取到元素,0个表 ...

  4. 第五天 Linux基本命令

    tty控制台终端  tty1~tty6? ctrl + alt + F2~F6  切换控制台 alt + F1 返回 但是使用 在图形化界面,使用init 3后,不能使用alt + F1返回,因为两者 ...

  5. 【转】ftrace 简介

    ftrace 简介 ftrace 的作用是帮助开发人员了解 Linux 内核的运行时行为,以便进行故障调试或性能分析. 最早 ftrace 是一个 function tracer,仅能够记录内核的函数 ...

  6. nyoj35——逆波兰表达式

    逆波兰表达式又称作后缀表达式,在四则混合运算的程序设计中用到. 例如: 1+2写成后缀表达式就是12+ 4+5*(3-2)的后缀表达式就是4532-*+ 后缀表达式在四则运算中带来了意想不到的方便,在 ...

  7. Math Issues

    Oh no, our Math object was "accidently" reset. Can you re-implement some of those function ...

  8. java程序设计基础篇 复习笔记 第七单元&&第八单元

    7.1 int[][] triArray{ {1}, {1,2}, {1,2,3}, }; 7.2 array[2].length 8.1 Unified Modeling Language:UML ...

  9. sql杂记

    Create procedure 存储过程的声明 PIVOT的一般语法是:PIVOT(聚合函数(列) FOR 列 in (…) )AS P 通俗简单的说:PIVOT就是行转列,UNPIVOT就是列传行 ...

  10. bfs+状态压缩dp

    题目连接 题解 : 对两两管道进行bfs,然后用dp[i][j] 来表示在i状态下通过了前j个管道 参考博客 #include<bits/stdc++.h> using namespace ...