异步和多线程的概念很容易混淆,很多人误认为异步方法一定在新的线程执行,有时调用异步方法时确实在新的线程,这里分几种情况,我们在本文中分析一下(获取程序当前执行线程ID,标示是否启动了新线程):
1、控制台程序调用自己写的异步函数
private static async Task<int> GetCantd(int x)
{
int sumall = 0;
Console.WriteLine("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
for (int i = 0; i < x; i++)
{
sumall += i;
}
return sumall;
}
控制台Main函数调用GetCantd函数并打印线程ID:
Console.WriteLine("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
Console.WriteLine(GetCantd(9999).ToString());
Console.WriteLine("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
可以看出虽然定义的是异步函数但在执行前、中、后,均为且含线程。
2、Winform程序调用自己写的异步函数
private StringBuilder Threadid=new StringBuilder();
private async Task<int> GetCantd(int x)
{
int sumall = 0;
Threadid.Append("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
// Console.WriteLine("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
for (int i = 0; i < x; i++)
{
sumall += i;
}
return sumall;
}
private void button1_Click(object sender, EventArgs e)
{
Threadid.Append("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
int count= GetCantd(50000).Result;
Threadid.Append("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
MessageBox.Show(Threadid.ToString());
}
可以看出虽然定义的是异步函数但在执行前、中、后,均为且含线程。
如何让异步跑在新的线程中:
3、让异步函数中执行的内容放在Task.Run() 新线程中执行:
private async Task<int> GetCantd(int x)
{
return await Task.Run(() =>
{
int sumall = 0;
Threadid.Append(" async id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
// Console.WriteLine("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
for (int i = 0; i < x; i++)
{
sumall += i;
}
return sumall;
});
}
private void button1_Click(object sender, EventArgs e)
{
Threadid.Append("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
GetCantd(500);
Threadid.Append("id:" + Thread.CurrentThread.ManagedThreadId + "\r\n");
MessageBox.Show(Threadid.ToString());
}
可以看到异步操作被切换到线程5执行。
问:此例中存在疑问,当我用int temp=GetCantd(500).result;接收执行结果时,点击button后界面一直在等待,且不会弹出messagebox,是什么原因呢 ?
4、在控制台中调用该函数的到相同的结果:
5、有时调用.net Framework 库函数时,部分异步函数被封装在新线程中执行,此种异步方法会让调用者误解,以为异步方法会被放在新线程中执行。
结论:异步方法不会自动创建新线程去处理,所以异步不等于是多线程,如果想让异步运行在新线程,需使用Task.Run()或Thread等。