咨询区
- soleiljy 
假设我们有一个涉及IO操作的方法 (读取数据库),这个方法支持以同步或者异步的方式执行。
- 同步方式 
IOMethod()
- 异步方式 
BeginIOMethod() EndIOMethod()
接下来我都用 Task 来包装这两个方法。
public static void Main(){var task1 = Task.Factory.StartNew(() => { IOMethod(); });task1.Wait();var task2 = Task.Factory.FromAsync(BeginIOMethod, EndIOMethod, ... );task2.Wait();}从资源利用率这个角度看,它们到底有什么不同?
回答区
- svick 
我就从第一种说起吧。
var task = Task.Factory.StartNew(() => { IOMethod(); });
task.Wait();你这种写法会阻塞两个线程:
- 调用线程 
这个很好理解,因为你用了 task.Wait() 这个阻塞版本。
- 线程池线程 
IOMethod 方法最终会被安排到线程池工作线程上,一直到该方法执行完毕。
接下来我们聊一下第二种方式:
var task = Task.Factory.FromAsync(BeginIOMethod, EndIOMethod, ... );
task.Wait();这种写法会阻塞一个线程
- 调用线程 
这个很好理解,你用了 Wait() 方法。
这里我有点疑惑,既然你都用了异步IO,为何还要用 Wait() 方法呢?可以将 Wait() 改成 await 关键词,从而实现 0 阻塞,这不是更好吗?比如下面这样:
var task = Task.Factory.FromAsync(BeginIOMethod, EndIOMethod, ... );
await task;当然如果你的项目是 .NET 4.0 的话,可以用 ContinueWith 替代,同样也可以实现 0 阻塞。
var task = Task.Factory.FromAsync(BeginIOMethod, EndIOMethod, ... );
task.ContinueWith(() => /* rest of the method here */);点评区
我感觉 svick 大佬介绍的非常到位,入木三分,如果再辅以 windbg 看看 threadpool 的情况会更好,学习了。