C#中的Task.Delay()延迟与异步执行返回结果
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了C#中的Task.Delay()延迟与异步执行返回结果,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含5576字,纯文字阅读大概需要8分钟。
内容图文
![C#中的Task.Delay()延迟与异步执行返回结果](/upload/InfoBanner/zyjiaocheng/647/0f411c39a109435fad3b918bb8289b47.jpg)
1.暂停一段时间
public static Task ShortDelay(TimeSpan delay)
{
await Task.Delay(delay);
Console.WriteLine(string.Format("延迟{0}", delay));
}
?
解析:
- 异步编程使用async与await关键字,搭配返回Task或其泛型
- async的存在是为了代码中await的生效
- 如果没有返回值,更推荐写为Task,而不是void
- Task.Delay()是异步编程提供的延迟方法,如果你想延迟两秒,可以Task.Delay(2000);
- 当Task.Delay(delay);执行后,会异步延迟delay的时间,在延迟的同时,执行下方的Console
- 当这行代码前+await,会等待异步延迟的执行结束后,执行下方的Console
2.实现简单的指数退避策略
pulic static async Task ToDoRetries()
{
var nextDelay = TimeSpan.FromSeconds(1);
for(int i = 0; i<3; i++)
{
return await DoSometingAsync();
nextDelay += nextDelay;
await Task.Delay(nextDelat);
}
}
?
解析:
- 退避策略:防止被访问的频繁被阻塞
3.实现超时处理
private static async Task<string> ToDoAsync()
{
await Task.Delay(TimeSpan.FromSeconds(3));
return "To Do Success!";
}
public static async Task<string> ToDoWithTimeOut()
{
var toDoTask = ToDoAsync();
var timeOutTask = Task.Delay(TimeSpan.FormSeconds(3));
var completedTask = await Task.WhenAny(toDoTask, timeOutTask);
if(completedTask == timeOutTask)
{
return "";
}
return await toDoTask;
}
?
解析:
- Task.WhenAny(toDoTask, timeOutTask);是只要其中有一个异步,或者说是任务完成,就返回
- 最费解就是最后最后一行,toDoTask是一个Task,并不是异步方法,为什么前面还有await,当将await去掉后:
- 断点后发现,toDoTask是一个Task类型,其中有id,status等属性,如果在Task类型前+await,获取的是Task类型中Result属性的值
出处:https://blog.csdn.net/PanPen120/article/details/53074627
===============================================================================================================
C#中的Task.Delay()和Thread.Sleep()
- Thread.Sleep()是同步延迟,Task.Delay()是异步延迟。
- Thread.Sleep()会阻塞线程,Task.Delay()不会。
- Thread.Sleep()不能取消,Task.Delay()可以。
- Task.Delay()实质创建一个运行给定时间的任务,Thread.Sleep()使当前线程休眠给定时间。
- 反编译Task.Delay(),基本上讲它就是个包裹在任务中的定时器。
- Task.Delay()和Thread.Sleep()最大的区别是Task.Delay()旨在异步运行,在同步代码中使用Task.Delay()是没有意义的;在异步代码中使用Thread.Sleep()是一个非常糟糕的主意。通常使用await关键字调用Task.Delay()。
- 我的理解:Task.Delay(),async/await和CancellationTokenSource组合起来使用可以实现可控制的异步延迟。
参考资料:
https://www.cnblogs.com/yy1234/p/8073732.html
https://blog.csdn.net/shu19880720/article/details/72901876
https://code.msdn.microsoft.com/ThreadSleep-vs-TaskDelay-766b46b7/view/Discussions#content
https://blog.csdn.net/wushang923/article/details/41015063
https://oomake.com/question/5779232
https://walterlv.com/post/sleep-delay-zero-vs-yield.html
以下是本人调试时的代码:
代码1:
using System; using System.Threading; using System.Threading.Tasks; namespace Delay_And_Sleep { class Program { static void Main(string[] args) { Task.Factory.StartNew(delegate { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ****** 开始Sleep()"); for (int i = 1; i < 20; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ***Sleep*** " + i); Thread.Sleep(100); } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ****** 结束Sleep()"); }); Task.Factory.StartNew(() => { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ====== 开始Delay()"); for (int i = 101; i < 120; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ===Delay=== " + i); Task.Delay(100);//需要.net4.5及以上 } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ====== 结束Delay()"); }); //Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + "press enter to close . . ."); Console.ReadLine(); } } }
运行结果:
代码2:
using System; using System.Threading.Tasks; namespace Delay_async_await { class Program { //该段代码通过async/awatit实现“同步”Delay static void Main(string[] args) { Task.Factory.StartNew(async () => { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ====== 开始Delay()"); for (int i = 101; i < 120; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ===Delay=== " + i); await Task.Delay(100);//需要.net4.5及以上 } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ====== 结束Delay()"); }); //Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + "press enter to close . . ."); Console.ReadLine(); } } }
运行结果:
代码3:
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; namespace 取消Delay { public partial class Form1 : Form { CancellationTokenSource cts = new CancellationTokenSource(); public Form1() { InitializeComponent(); } void PutThreadSleep() { Thread.Sleep(5000); } async Task PutTaskDelay() { try { await Task.Delay(5000, cts.Token);//需要.net4.5的支持 } catch (TaskCanceledException ex) { MessageBox.Show(ex.ToString()); } } private void btnThreadSleep_Click(object sender, EventArgs e) { PutThreadSleep(); MessageBox.Show("Sleep : I am back"); } //使用async/await便于观察效果;不用的话就直接弹出MessageBox了 private async void btnTaskDelay_Click(object sender, EventArgs e) { await PutTaskDelay(); MessageBox.Show("Delay : I am back"); } private void btnCancelTaskDelay_Click(object sender, EventArgs e) { cts.Cancel(); } } }
运行结果:
出处:https://blog.csdn.net/chenweicode/article/details/91372281
========================================================================================
内容总结
以上是互联网集市为您收集整理的C#中的Task.Delay()延迟与异步执行返回结果全部内容,希望文章能够帮你解决C#中的Task.Delay()延迟与异步执行返回结果所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。