C#中的Task.Delay()延迟与异步执行返回结果

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去掉后:
    C#中的Task.Delay()延迟与异步执行返回结果
  • 断点后发现,toDoTask是一个Task类型,其中有id,status等属性,如果在Task类型前+await,获取的是Task类型中Result属性的值

 

 

出处:https://blog.csdn.net/PanPen120/article/details/53074627

===============================================================================================================

 

        private async void btTaskTest_Click(object sender, EventArgs e)
        {
            //Task<string> retStr = ToDoWithTimeOut();
            var res = await ToDoWithTimeOut();
            label4.Text = retStr.Result;
            label4.Text = res;
        }

 

===============================================================================================================

C#中的Task.Delay()和Thread.Sleep() 

  1. Thread.Sleep()是同步延迟,Task.Delay()是异步延迟。
  2. Thread.Sleep()会阻塞线程,Task.Delay()不会。
  3. Thread.Sleep()不能取消,Task.Delay()可以。
  4. Task.Delay()实质创建一个运行给定时间的任务,Thread.Sleep()使当前线程休眠给定时间。
  5. 反编译Task.Delay(),基本上讲它就是个包裹在任务中的定时器。
  6. Task.Delay()和Thread.Sleep()最大的区别是Task.Delay()旨在异步运行,在同步代码中使用Task.Delay()是没有意义的;在异步代码中使用Thread.Sleep()是一个非常糟糕的主意。通常使用await关键字调用Task.Delay()。
  7. 我的理解: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

http://social.technet.microsoft.com/wiki/contents/articles/21177.visual-c-thread-sleep-vs-task-delay.aspx(评论区有争议)

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();
        }
    }
}

 

运行结果:

C#中的Task.Delay()延迟与异步执行返回结果

代码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();
        }
    }
}

 

运行结果:

C#中的Task.Delay()延迟与异步执行返回结果

代码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();
        }
    }
}

 

运行结果:

C#中的Task.Delay()延迟与异步执行返回结果

 

 

出处:https://blog.csdn.net/chenweicode/article/details/91372281

========================================================================================

C#中的Task.Delay()延迟与异步执行返回结果

上一篇:zabbix笔记之Windows性能监视器


下一篇:WPF 中动态创建和删除控件