async/await的多线程问题
今天尝试把.net4.5新增的异步编程模型async/await加入自己的框架,因为从第一印象看,使用async/await的写法实在太方便了,以同步代码的方式写异步流程,写起来更顺畅,不容易打断思路,异常传递、资源控制(lock和using)也都完美支持,即使有少量的性能损失,也完全可以接受。
首先我写了一个测试代码,以熟悉async/await模型,代码如下:
static class TestCase
{
static async Task Test2()
{
Console.WriteLine("3 {0}", Thread.CurrentThread.ManagedThreadId);
await Task.Run(() =>
{
Thread.Sleep();
});// 这里可以换成Task.Delay(1000);
Console.WriteLine("4 {0}", Thread.CurrentThread.ManagedThreadId);
} static async Task Test1()
{
Console.WriteLine("1 {0}", Thread.CurrentThread.ManagedThreadId);
await Test2();
Console.WriteLine("2 {0}", Thread.CurrentThread.ManagedThreadId);
} static void Main(string[] args)
{
Test1().Wait();
}
}
输出如下:
代码很简单,结果也看似正确,直到我发现一个大问题:
await后面的代码是在另一个线程执行的!!!
这不仅会影响到框架的运行,还会导致线程争用资源的问题,也就是说看似同一处的代码,会运行在不同线程,并且有可能并行。如果访问公共资源(如静态变量)还需要加锁,严重影响编程体验和性能。我推测,很可能Task类自带的创建Task的静态函数所产生的Task,设置Complete状态时都是在其他线程,因此导致了await的回调也在该线程执行。于是我把代码稍微改了一下:
static class TestCase
{
static TaskCompletionSource<object> source = new TaskCompletionSource<object>(); static async Task Test2()
{
Console.WriteLine("3 {0}", Thread.CurrentThread.ManagedThreadId);
await source.Task;
Console.WriteLine("4 {0}", Thread.CurrentThread.ManagedThreadId);
} static async Task Test1()
{
Console.WriteLine("1 {0}", Thread.CurrentThread.ManagedThreadId);
await Test2();
Console.WriteLine("2 {0}", Thread.CurrentThread.ManagedThreadId);
} static void Main(string[] args)
{
Task task = Test1();
Thread.Sleep();
source.SetResult(null);
task.Wait(); // 这一句其实是没有必要的,并且如果放在SetResult前,会导致死锁
}
}
输出如下:
果然全部是在主线程执行了。其中source.SetResult(null);这一句放到框架中,当条件合适时就在主线程Loop中调用,以便唤醒await继续执行剩下的过程。此外,Task类的静态函数所产生的Task,也可以通过一个包装函数,来让在其他线程执行的SetResult,Queue到主线程调用,类似这样:
public Task<T> Wrap<T>(Task<T> task)
{
Loop loop = Current;
TaskCompletionSource<T> source = new TaskCompletionSource<T>();
task.GetAwaiter().OnCompleted(() =>
{
loop.Execute(() =>
{
if (task.IsCompleted)
source.TrySetResult(task.Result);
else if (task.IsCanceled)
source.TrySetCanceled();
else
source.TrySetException(task.Exception);
});
});
return source.Task;
}
async/await的多线程问题的更多相关文章
- C# Async/await 异步多线程编程
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.N ...
- 多线程(5)async&await
.net 4.0的Task已经让我们可以非常简单地使用多线程,并且可以有返回值,也可以支持线程的取消等操作,可谓已经很强大了.但.net 4.5为我们带来了async&await,使得实现多线 ...
- async await 多线程
async await 并没有开启多线程 出现await的地方 只是开启了一个子线程继续往后执行 主线程返回 防止阻塞 相当于 await customerRepository.getall() ...
- 聊聊多线程那一些事儿 之 五 async.await深度剖析
hello task,咱们又见面啦!!是不是觉得很熟读的开场白,哈哈你哟这感觉那就对了,说明你已经阅读过了我总结的前面4篇关于task的文章,谢谢支持!感觉不熟悉的也没有关系,在文章末尾我会列出前四 ...
- async/await到底该怎么用?如何理解多线程与异步之间的关系?
前言 如标题所诉,本文主要是解决是什么,怎么用的问题,然后会说明为什么这么用.因为我发现很多萌新都会对之类的问题产生疑惑,包括我最初的我,网络上的博客大多知识零散,刚开始看相关博文的时候,就这样.然后 ...
- C#多线程和异步(二)——Task和async/await详解
一.什么是异步 同步和异步主要用于修饰方法.当一个方法被调用时,调用者需要等待该方法执行完毕并返回才能继续执行,我们称这个方法是同步方法:当一个方法被调用时立即返回,并获取一个线程执行该方法内部的业务 ...
- C#多线程和异步(二)——Task和async/await详解(转载)
一.什么是异步 同步和异步主要用于修饰方法.当一个方法被调用时,调用者需要等待该方法执行完毕并返回才能继续执行,我们称这个方法是同步方法:当一个方法被调用时立即返回,并获取一个线程执行该方法内部的业务 ...
- C#多线程和异步——Task和async/await详解
阅读目录 一.什么是异步 二.Task介绍 1 Task创建和运行 2 Task的阻塞方法(Wait/WaitAll/WaitAny) 3 Task的延续操作(WhenAny/WhenAll/Cont ...
- [.NET] 利用 async & await 的异步编程
利用 async & await 的异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/5922573.html 目录 异步编程的简介 异 ...
随机推荐
- Giraph入门
概要 这是一个Giraph的入门教程,主要用来运行少量输入的Giraph程序,并不能用于生产环境. 在这个教程中,我们将会在一个物理机器行部署一个单节点,伪分布的Hadoop集群.这个节点既是mast ...
- [Hadoop] - SSH免密码登录
在安装hadoop之前需要进行ssh免密码登录,ssh 无密码登录要使用公钥与私钥.linux下可以用用ssh-keygen生成公钥/私钥对,下面我以Redhat为例. 我这里只采用一台机器A(10. ...
- 做一款直播类app
ijkplayer 是一款做视频直播的框架, 基于ffmpeg, 支持 Android 和 iOS, 网上也有很多集成说明, 但是个人觉得还是不够详细, 在这里详细的讲一下在 iOS 中如何集成ijk ...
- java中 i = i++ 的结果
昨天看到下面这段代码,分享出来给大家看看,大家也可以讨论讨论. int i = 0; i = i++; System.out.println("i的值是 "+i); 根据我们通常所 ...
- MySQL调优三步曲(慢查询、explain profile)
在做性能测试中经常会遇到一些sql的问题,其实做性能测试这几年遇到问题最多还是数据库这块,要么就是IO高要么就是cpu高,所以对数据的优化在性能测试过程中占据着很重要的地方,下面我就介绍一些msyql ...
- 【Harmony】概述
原文来自本人的微信公众号文章 系统工程实验室 引言 基于模型的系统工程(简称MBSE,英文全称Model based System Engineering )的实践至少需要三个维度的支撑 ...
- TableView 多余分割线的处理
方法一,以下两个方法的实现 - (void)viewDidLoad { [super viewDidLoad]; self.tableView.tableFooterView = [[UIView a ...
- 2015年ACM-ICPC亚洲区域赛合肥站网络预选赛H题——The Next (位运算)
Let L denote the number of 1s in integer D's binary representation. Given two integers S1 and S2, we ...
- github 删除远程仓库项目中的任意文件夹
今天上传代码把不需要的push上去了.结果想删除那个不想要的怎么弄都不行.网上大部分都是把那个项目整个暴力删除.那可不行啊那么多都删除.下次上传不是要命啊! 试啊试终于解决了.顺便记录一下也帮助下需要 ...
- 转换器4:手写PHP转Python编译器,语法解析部分
写完词法部分,又有很多杂事,周末终于有空来实现伟大的语法解析部分了. 撸完代码之后发现,程序太短了,不算上状态机,才186行(含注释),关键代码不到100行.运行调试过后,发现还行.居然可以解析One ...