前面两篇回顾线程和线程池的使用方法,微软在.NET4.5推出了新的线程模型-Task。本篇将简单的介绍Task的使用方法。

Task与线程

  Task与线程或者说线程池关系紧密,可以说是基于线程池实现的,虽说任务最终还是要抛给线程去执行,但是Task仍然会比线程、线程池的开销要小,并且提供了可靠的API来控制线任务执行。

  使用Task来执行的任务最终会交给线程池来执行,若该任务需要长时间执行,可以将其标记为LongRunning,这是便会单独去请求创建线程来执行该任务。

Task

 创建

  Task的创建也存在两种方式,使用new或者使用静态工厂方式来创建:

        static void Main(string[] args)
{ Task t = new Task(Menthod1);
t.Start(); Task.Factory.StartNew(Menthod2); Console.WriteLine("主线程的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("--------------------");
Console.ReadLine();
} static void Menthod1()
{
Thread.Sleep();
Console.WriteLine("线程1的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("--------------------");
}
static void Menthod2()
{
Thread.Sleep();
Console.WriteLine("线程2的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("--------------------");
}

  上面说到使用Task执行任务最终还是需要线程池来执行,若不想让线程池来执行,可以添加LongRunning标志:

        Task t = new Task(Menthod1, TaskCreationOptions.LongRunning);
t.Start(); Task.Factory.StartNew(Menthod2, TaskCreationOptions.LongRunning);

  TaskCreationOptions还有很多枚举值,用来控制任务的更多属性。

 参数与返回值

  使用Task也可以传入参数(object类型)与返回值:

        static void Main(string[] args)
{ Task.Factory.StartNew(Menthod1, );
//int result = Task.Factory.StartNew(new Func<object, int>(Menthod3), 233).Result;
Task<int> t = Task.Factory.StartNew(new Func<object, int>(Menthod3), );
int result = t.Result; Console.WriteLine("主线程的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("result:{0}", result);
Console.WriteLine("--------------------");
Console.ReadLine();
} static void Menthod1(object obj)
{
Thread.Sleep();
Console.WriteLine("线程1的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("obj:{0}", obj);
Console.WriteLine("--------------------");
}
static int Menthod3(object obj)
{
Thread.Sleep();
Console.WriteLine("线程3的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("obj:{0}", obj);
Console.WriteLine("--------------------");
return obj;
}

  这里为了获取返回值,阻塞了主线程,这里请注意一下。

 等待Task

  可以使用Task实例的Wait方法来实现等待任务结束,也可以向多线程一样,使用WaitAll和WaitAny一样来等待多个任务结束,不过操作更为简单:

        t.Wait();
Task.WaitAll(t1, t2 ...);
Task.WaitAny(t1, t2 ...);

 取消Task

  任务也是可以事先取消的,不过需要使用CancellationTokenSource:

        static void Main(string[] args)
{
Task.Factory.StartNew(Menthod1); CancellationTokenSource cancelTokenSource = new CancellationTokenSource();
Task.Factory.StartNew(Menthod2, cancelTokenSource.Token);
cancelTokenSource.Cancel(); Console.WriteLine("主线程的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("--------------------");
Console.ReadLine();
}

  上述代码就可以将线程2给取消掉,当然,Cancel方法可以自己找个合适的地方调用。

 继续Task

  在Task中,可以实现在一个任务结束后开启另一个新的任务:

        static void Main(string[] args)
{
Task t = Task.Factory.StartNew(Menthod1);
t.ContinueWith(new Action<Task>(Menthod4)); Console.WriteLine("主线程的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("任务t的ID:{0}", t.Id);
Console.WriteLine("--------------------");
Console.ReadLine();
} static void Menthod1()
{
Thread.Sleep();
Console.WriteLine("线程1的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("--------------------");
}
static void Menthod4(Task t)
{
Console.WriteLine("线程4的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("任务t的ID:{0}", t.Id);
Console.WriteLine("--------------------");
}

  这里因为使用了委托,也可以使用lambda表达式,更简单一些调用:

        Task t = Task.Factory.StartNew(Menthod1);
t.ContinueWith(task =>
{
Console.WriteLine("线程4的ID:{0}", Thread.CurrentThread.ManagedThreadId);
Console.WriteLine("任务t的ID:{0}", task.Id);
Console.WriteLine("--------------------");
});

总结

  使用传统线程方式来进行多线程编程的时候,对线程的控制总是不到位,产生一些奇奇怪怪的问题;或是代码写得很杂乱;或是开发人员乱用线程,比方说无限制的创建线程、将线程池线程占满,等等。

  Task的出现,实现对传统线程操作的封装,提供可靠高效的API来控制线程,极大的方便多线程编程,所以可以用到Task的地方尽量使用Task;当然,这里仍会产生线程安全的问题,同样需要进行线程同步,与上一篇处理方式类似。

  Task的内容还有很多,更深层的运行原理、封装还等着我们去了解,现在只把我用到的东西分享出来,希望能帮助到需要的人。

  线程的学习就先告一段落了,敬请期待新的内容吧。

菜鸟之旅——学习线程(Task)的更多相关文章

  1. [C#][算法] 用菜鸟的思维学习算法 -- 马桶排序、冒泡排序和快速排序

    用菜鸟的思维学习算法 -- 马桶排序.冒泡排序和快速排序 [博主]反骨仔 [来源]http://www.cnblogs.com/liqingwen/p/4994261.html  目录 马桶排序(令人 ...

  2. C#多线程之旅(3)——线程池

    v博客前言 先交代下背景,写<C#多线程之旅>这个系列文章主要是因为以下几个原因:1.多线程在C/S和B/S架构中用得是非常多的;2.而且多线程的使用是非常复杂的,如果没有用好,容易造成很 ...

  3. 上四条只是我目前总结菜鸟们在学习FPGA时所最容易跑偏的地

    长期以来很多新入群的菜鸟们总 是在重复的问一些非常简单但是又让新手困惑不解的问题.作为管理员经常要给这些菜鸟们普及基础知识,但是非常不幸的是很多菜鸟怀着一种浮躁的心态来学习 FPGA,总是急于求成. ...

  4. Java核心知识点学习----线程中如何创建锁和使用锁 Lock,设计一个缓存系统

    理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...

  5. C#中假设正确使用线程Task类和Thread类

    C#中使用线程Task类和Thread类小结 刚接触C#3个月左右.原先一直使用C++开发.由于公司的须要,所地採用C#开发.主要是控制设备的实时性操作,此为背景. 对于C#中的Task和Thread ...

  6. 菜鸟教程之学习Shell script笔记(下)

    菜鸟教程Shell script学习笔记(下) 以下内容是学习菜鸟教程之shell教程,所整理的笔记 菜鸟教程之shell教程:http://www.runoob.com/linux/linux-sh ...

  7. 菜鸟教程之学习Shell script笔记(上)

    菜鸟教程之学习Shell script笔记 以下内容是,学习菜鸟shell教程整理的笔记 菜鸟教程之shell教程:http://www.runoob.com/linux/linux-shell.ht ...

  8. Hadoop源码学习笔记(3) ——初览DataNode及学习线程

    Hadoop源码学习笔记(3) ——初览DataNode及学习线程 进入了main函数,我们走出了第一步,接下来看看再怎么走: public class DataNode extends Config ...

  9. Python学习---线程/协程/进程学习 1220【all】

    Python学习---线程基础学习 Python学习---线程锁/信号量/条件变量同步1221 Python学习---同步条件event/队列queue1223 Python学习---进程 1225 ...

随机推荐

  1. Android--app性能问题的总结(一)

     一个应用程序的性能问题体现在很多方面,app的性能问题,很大程度上决定了使用app的用户量,如果正在使用app的过程中出现app崩溃.卡顿半天加载不出数据(跟网络也有一定的关系).用户请求事件半天获 ...

  2. python——几种截图对比方式!

    本次记录的几种截图对比方式,主要是为了在进行手机自动化测试时,通过截图对比来判断测试的正确性,方式如下: # -*- coding: utf- -*- ''' 用途:利用python实现多种方法来实现 ...

  3. ToB蓝海的台阶-PaaS,SaaS技术详解

    前言 随着大量SaaS公司进入市场,我们看到颠覆性的软件服务以各种方式进入企业流程-从营销工具到支付系统.随着SaaS帮助优化业务流程,实现更流畅和自动化的运营,风险投资公司首先潜入池中寻找最优秀和最 ...

  4. JAVA线程及简单同步实现的原理解析

    线程 一.内容简介: 本文主要讲述计算机中有关线程的相关内容,以及JAVA中关于线程的基础知识点,为以后的深入学习做铺垫.如果你已经是高手了,那么这篇文章并不适合你. 二.随笔正文: 1.计算机系统组 ...

  5. 流程控制之for循环

    目录 语法(掌握) for+break for+continue for循环嵌套(掌握) for+else(掌握) for循环实现loading(了解) 语法(掌握) 为什么有了while循环,还需要 ...

  6. Hadoop大数据部署

    Hadoop大数据部署 一. 系统环境配置: 1. 关闭防火墙,selinux 关闭防火墙: systemctl stop firewalld systemctl disable firewalld ...

  7. Asp.Net Core中HttpClient的使用方式

    在.Net Core应用开发中,调用第三方接口也是常有的事情,HttpClient使用人数.使用频率算是最高的一种了,在.Net Core中,HttpClient的使用方式随着版本的升级也发生了一些变 ...

  8. 服务端预渲染之Nuxt(爬坑篇)

    Nuxt是解决SEO的比较常用的解决方案,随着Nuxt也有很多坑,每当突破一个小技术点的时候,都有很大的成就感,在这段时间里着实让我痛并快乐着.在这里根据个人学习情况,所踩过的坑做了一个汇总和总结. ...

  9. SQLSERVER数据库死锁与优化杂谈

    死锁杂谈 当数据库死锁时,SqlServer会释放一个优先级较低的锁,让另一个事务运行:所以,即时去捕捉数据库死锁,是挺不容易的. 如果,数据库死锁比较长时间,那么死锁是可以被捕捉的. 可以用SqlS ...

  10. Android版数据结构与算法(一):基础简介

    版权声明:本文出自汪磊的博客,未经作者允许禁止转载. 一.前言 项目进入收尾阶段,忙忙碌碌将近一个多月吧,还好,不算太难,就是麻烦点. 数据结构与算法这个系列早就想写了,一是梳理总结,顺便逼迫自己把一 ...