国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

并行編程中的取消任務、共享狀態,等等

2019-11-14 14:11:16
字體:
來源:轉載
供稿:網友

 

在面對相互獨立的數據或者相互獨立的任務時,也許正是Parallel登場的時候。

 

比如說有一個盒子的集合,分別讓盒子旋轉一定的角度。

 

void RotateBox(IEnumerable<Box> boxes, float degree)
{
    Parallel.ForEach(boxes, box => box.Rotate(degree));
}

 

如果并行任務中的一個任務出現異常,需要結束出問題的任務呢?

 

Parallel.ForEach為我們提供了一個重載方法,可以控制任務是否繼續。

 

void RotateBoxes(IEnumerable<Box> boxes)
{
    Parallel.ForEach(boxes, (box, state) => {
        if(!box.IsInvertible)
        {
            state.Stop();
        }
        else
        {
            box.Invert();
        }
    });
}

 

如果想取消整個并行任務呢?

 

Parallel.ForEach也為我們提供一個重載方法,可以接收一個ParallelOption的形參,通過設置ParallelOption的CancellationToken屬性來取消整個并行過程。

 

void RotateBoxes(IEnumerable<Box> boxes,float degrees, CancellationToken token)
{
    Paralle.ForEach(boxes, 
    new ParallelOptions{CancellationToken=token},
    box => box.Rotate(degrees));
}

 

在使用的時候,一般先定義個全局CancellationTokenSource類型的變量。

 

static CancellationTokenSource token = new CancellationTokenSource();

 

然后,在某個并行任務中設置取消。

 

token.Cancel();

 

最后,再把這個token賦值給以上方法的CancellationToken屬性。

 

各個并行任務如何共享狀態,共享一個變量呢?

 

int InvertBoxes(IEnumerable<Box> boxes)
{
    object mutex = new object();//用來鎖
    int nonInvertibleCount = 0; //各任務共享的變量
    Paralle.ForEach(boxes, box =>{
        if(box.IsInvertible){
            box.Invert();
        }
        else
        {
            lock(mutex)
            {
                ++nonInvertibleCount;
            }
        }
    });
    return nonInvertibleCount;
}

 

可見,對于各并行線程共享的變量,需要加一個線程鎖,以防止多個線程同時操作共享變量。

 

另外,Parallel.ForEach提供了一個重載,其中localFinally形參接收一個委托類型,通過該委托讓并行任務共享一個變量。比如:

 

static int ParallelSum(IEnumerable<int> values)
{
    object mutex = new object();
    int result = 0;
    Parallel.ForEach(
        source: values,
        LocalInit: () => 0,
        body: (item, state, localVlaue) => localValue + item,
        localFinally: localValue => {
            lock(mutex)
            {
                result += localValue;
            }
        }
    );
    return result;
}

 

當然,也別忘了PLINQ也支持并行

 

static int ParalleSum(IEnumerable<int> values)
{
    return values.asparallel().Sum();
}

 

PLINQ的Aggregate方法也可實現:

 

static int ParallelSum(IEnumerable<int> values)
{
    return values.AsParallel().Aggregate(
        seed: 0,
        func: (sum, item) => sum + item;
    );
}

 

以上,是對相互獨立數據的處理。

 

那么,如何處理相互獨立的多個任務呢?

 

通過Parallel.Invoke方法可以實現。

 

static voiD PRocessArray(double[] array)
{
    Parallel.Invoke(
        () => ProcessPartialArray(array, 0, array.Length/2),
        () => ProcessPartialArray(array, array.Length/2, array.Length)
    );
}
static void ProcessPartialArray(double[] array, int begin, int end)
{}

 

使用Parallel.Invoke方法還可以讓一個Action或這方法執行多次。

 

static void Do20(Action action)
{
    //讓某個Action執行20次
    Action[] actions = Enumerable.Repeat(action, 20).ToArray();
    Parallel.Invoke(actions);
}

 

Parallel.Invoke方法也提供了重載,接收ParallelOption類型的形參,用來控制取消整個并行過程。

 

static void Do20(Action action)
{
    //讓某個Action執行20次
    Action[] actions = Enumerable.Repeat(action, 20).ToArray();
    Parallel.Invoke(new ParallelOptions{CancellationToken = token},actions);
}

 

參考資料:C#并發編程經典實例


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 商南县| 龙海市| 垦利县| 龙泉市| 屯留县| 章丘市| 绥化市| 县级市| 潞西市| 云龙县| 慈溪市| 许昌县| 航空| 铅山县| 左贡县| 光泽县| 甘肃省| 河源市| 平南县| 怀化市| 鹤峰县| 盐城市| 伊春市| 武强县| 南川市| 南汇区| 博客| 遂平县| 延吉市| 从化市| 景谷| 普宁市| 贞丰县| 兴安县| 济源市| 鹤峰县| 阿图什市| 阜平县| 深泽县| 慈利县| 南江县|