zoukankan      html  css  js  c++  java
  • Task.Run 和 Task.Factory.StartNew 区别

    Task.Run 是在 dotnet framework 4.5 之后才可以使用, Task.Factory.StartNew 可以使用比 Task.Run 更多的参数,可以做到更多的定制。

    可以认为 Task.Run 是简化的 Task.Factory.StartNew 的使用,除了需要指定一个线程是长时间占用的,否则就使用 Task.Run

    创建新线程

    下面来告诉大家使用两个函数创建新的线程

    Task.Run(() =>
    {
       var foo = 2;
    });

    这时 foo 的创建就在另一个线程,需要知道 Task.Run 用的是线程池,也就是不是调用这个函数就会一定创建一个新的线程,但是会在另一个线程运行。

    Task.Factory.StartNew(() =>
    {
        ar foo = 2;
    });

    可以看到,两个方法实际上是没有差别,但是Task.Run比较好看,所以推荐使用Task.Run

    等待线程

    创建的线程,如果需要等待线程执行完成在继续,那么可以使用 await 等待

    复制代码

    private static async void SeenereKousa()
    {
         Console.WriteLine("开始 线程"+Thread.CurrentThread.ManagedThreadId);
         await Task.Run(() =>
         {
             Console.WriteLine("进入 线程" + Thread.CurrentThread.ManagedThreadId);
         });
         Console.WriteLine("退出 线程"+Thread.CurrentThread.ManagedThreadId);
    }

    复制代码

    但是需要说的是这里使用 await 主要是给函数调用的外面使用,上面代码在函数里面使用 await 函数是 void 那么和把代码放在 task 里面是相同

    复制代码

    private static async void SeenereKousa()
    {
        Console.WriteLine("开始 线程"+Thread.CurrentThread.ManagedThreadId);
        await Task.Run(() =>
        {
             Console.WriteLine("进入 线程" + Thread.CurrentThread.ManagedThreadId);
             Console.WriteLine("退出 线程"+Thread.CurrentThread.ManagedThreadId);
        });
    }

    复制代码

    但是如果把 void 修改为 Task ,那么等待线程才有用

    除了使用 await 等待,还可以使用 WaitAll 等待

    复制代码

     Console.WriteLine("开始 线程" + Thread.CurrentThread.ManagedThreadId);
     var t = Task.Run(() =>
     {
          Console.WriteLine("进入 线程" + Thread.CurrentThread.ManagedThreadId);
     });
    
     Task.WaitAll(t);
     Console.WriteLine("退出 线程" + Thread.CurrentThread.ManagedThreadId);

    复制代码

    使用 WaitAll 是在调用 WaitAll 的线程等待,也就是先在线程 1 运行,然后异步到 线程2 运行,这时线程1 等待线程2运行完成再继续,所以输出

    1.  
      开始 线程1
    2.  
      进入 线程2
    3.  
      退出 线程1

    长时间运行

    两个函数最大的不同在于 Task.Factory.StartNew 可以设置线程是长时间运行,这时线程池就不会等待这个线程回收

    复制代码

    Task.Factory.StartNew(() =>
    {
          for (int i = 0; i < 100; i++)
          {
               var foo = 2;
          }
          Console.WriteLine("进行 线程" + Thread.CurrentThread.ManagedThreadId);
    }, TaskCreationOptions.LongRunning);

    复制代码

    所以在需要设置线程是长时间运行的才需要使用 Task.Factory.StartNew 不然就使用 Task.Run

    调用 Task.Run(foo) 就和使用下面代码一样

    Task.Factory.StartNew(foo, CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default);

    最后再说明下await和async与Task的联系:
    async标记会告诉编辑器接下来的方法可能会用到异步,当然并不一定会有用到,一直向下到await这是会等待,await 不会开启新的线程,
    当前线程会一直往下走直到遇到真正的Async方法(比如说HttpClient.GetStringAsync),这个方法的内部会用Task.Run或者Task.Factory.StartNew 去开启线程(因为task底层是线程池机制,也可能是复用线程)。
    如果方法不是.NET为我们提供的Async方法,我们需要自己创建Task,才会真正的去创建线程。
  • 相关阅读:
    带掩码的自编码器MAE详解和Pytorch代码实现
    联邦学习(Federated Learning)详解以及示例代码
    SIMILAR:现实场景中基于子模块信息度量的主动学习
    BERT 模型的知识蒸馏: DistilBERT 方法的理论和机制研究
    为什么 Pi 会出现在正态分布的方程中?
    快到周五了
    土豆
    忙碌的周末
    周五了
    写给妹妹的祝福语
  • 原文地址:https://www.cnblogs.com/wangyu19900123/p/11811312.html
Copyright © 2011-2022 走看看