.net的简易多线程处理

这篇文章是对几年前写的<Task及其异常处理的若干事项>的一些狗尾续貂的补充。

更简单的写法

几年前写的那篇文章很详细地描述了.net用Task对线程进行封装的相关技术。开一个新的线程去执行一个任务,当时是这么写的:

Task taskA = Task.Factory.StartNew(() =>{
    //Do something...
});

现在还有一种更简单的写法:

Task.Run(() => {
    //Do something...
});

延迟执行

另外还有“延迟执行”,啥叫延迟执行?就是:多少时间后,给我执行这个动作!

Task.Delay(mSec).ContinueWith(_ => {
    //Do something...
});

 延迟执行后在UI线程中执行某个操作(通常是更新界面)。

Task.Delay(mSec).ContinueWith(_ => {
    //Do UI update operation
}, TaskScheduler.FromCurrentSynchronizationContext());

UI更新

这样开线程执行任务的话是不能操作UI上的元素,那如何操作UI上的元素?(以WPF为例)

Dispatcher.BeginInvoke((Action)(() => {
    //Do UI operation here...
}));

闭包(Closure)

另外还可以实现“闭包”,即Task这段代码访问它外部的变量:

int value = 10;
Task.Run(() => {
    Debug.WriteLine("value is " + value);
});

说到闭包,要记住这里有个坑:

for (int i = 0; i < 10; i++) {
    Task.Run(() => {
        Debug.WriteLine("i value is " + i);
    });
}

输出全是10,想输出1-9,有两种办法,1是赋值一个局部变量:

for (int i = 0; i < 10; i++) {
    int j=i;
    Task.Run(() => {
        Debug.WriteLine("i value is " + j);
    });
}

另一种方法是用foreach:

int[] arr = {0,1,2,3,4,5,6,7,8,9};
foreach (int i in arr) {
    Task.Run(() => {
        Debug.WriteLine("i value is " + i);
    });
}
posted @ 2016-05-03 16:29  guogangj  阅读(1419)  评论(0编辑  收藏  举报