Task和Task<TResult>是c#提供的一种实现异步功能的2个类。Task<TResult>继承Task类,有返回参数。
1、基本用法 不嵌套
利用静态方法创建和运行任务:
//无返回参数,Action委托作为参数。有参数,无返回值 Task.Run( ()=>{ //Action委托 Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作."); }); //有返回参数,func委托作为参数。有参数,有返回值 Task<string> t=Task.Run<string>(()=>{ //Func<TResult>委托 string str="有参任务" return str; }); //开启另一个任务等待完成,并输出结果 Task cwt = t.ContinueWith(m => { Console.WriteLine("任务完成后的执行结果:{0}", m.Result.ToString()); //Action<Task<TResult>>委托,m代表mycaller即调用者,在这里m=t. });
利用构造函数创建和执行任务:
//此处只说明有返回值情况 Task<int> t = new Task<int>(() => { int sum=0; Console.WriteLine("使用System.Threading.Tasks.Task执行异步操作."); for (int i = 0; i < 10; i++) { sum =+ i; } return sum; }); //启动任务,并安排到当前任务队列线程中执行任务(System.Threading.Tasks.TaskScheduler) t.Start(); //等待任务的完成执行过程。这里有两种方法,一种,使用已创建任务的Wait方法。会在这里形成阻塞 t.Wait(); 【另一种不会形成阻塞的方法:使用已创建任务的ContinueWith方法。作用:任务完成后,再开启另外一个任务,对结果进行处理。 Task cwt = t.ContinueWith(m => { Console.WriteLine("任务完成后的执行结果:{0}", m.Result.ToString()); }); 】 //获得任务的执行结果 Console.WriteLine("任务执行结果:{0}", t.Result.ToString());
2、任务嵌套。父任务,子任务。
通过Task类创建的任务是顶级任务,可以通过使用 TaskCreationOptions.AttachedToParent 标识把这些任务与创建他的任务相关联,所有子任务全部完成以后父任务才会结束操作。示例如下:
1: static void ParentChildTask() { 2: Task<string[]> parent = new Task<string[]>(state => { 3: Console.WriteLine(state); 4: string[] result=new string[2]; 5: //创建并启动子任务 6: new Task(() => { result[0]= "我是子任务1。"; },TaskCreationOptions.AttachedToParent).Start(); 7: new Task(() => { result[1] = "我是子任务2。"; }, TaskCreationOptions.AttachedToParent).Start(); 8: return result; 9: },"我是父任务,并在我的处理过程中创建多个子任务,所有子任务完成以后我才会结束执行。"); 10: //任务处理完成后执行的操作 11: parent.ContinueWith(t => { 12: Array.ForEach(t.Result, r=>Console.WriteLine(r)); 13: }); 14: //启动父任务 15: parent.Start(); 16: Console.Read(); 17: }
如果需要创建一组具有相同状态的任务时,可以使用TaskFactory类或TaskFactory<TResult>类。这两个类创建一组任务时可以指定任务的CancellationToken、TaskCreationOptions、TaskContinuationOptions和TaskScheduler默认值。示例代码:
1: static void TaskFactoryApply() 2: { 3: Task parent = new Task(() => 4: { 5: CancellationTokenSource cts = new CancellationTokenSource(5000); 6: //创建任务工厂 7: TaskFactory tf = new TaskFactory(cts.Token, TaskCreationOptions.AttachedToParent, TaskContinuationOptions.ExecuteSynchronously, TaskScheduler.Default); 8: //添加一组具有相同状态的子任务 9: Task[] task = new Task[]{ 10: tf.StartNew(() => { Console.WriteLine("我是任务工厂里的第一个任务。"); }), 11: tf.StartNew(() => { Console.WriteLine("我是任务工厂里的第二个任务。"); }), 12: tf.StartNew(() => { Console.WriteLine("我是任务工厂里的第三个任务。"); }) 13: }; 14: }); 15: parent.Start(); 16: Console.Read(); 17: }