Опубликован: 15.10.2009 | Уровень: специалист | Доступ: платный
Лекция 9:

Примеры программирования с использованием библиотеки PFX

< Лекция 8 || Лекция 9: 12 || Лекция 10 >
Аннотация: В данной лекции будет представлен ряд примеров программирования с использованием механизма задач и некоторых конструкций из библиотеки PFX, связанных с ними.

В предыдущих лекциях были рассмотрены различные классы библиотеки PFX и способы работы с ними. В данном разделе будет представлен ряд примеров программирования с использованием механизма задач и некоторых конструкций из библиотеки PFX, связанных с ними.

9.1 Реализация конструкций ContinueWhenAll и ContinueWhenAny

Библиотека PFX, а именно, ее составная часть Task Parallel Library (TPL), предоставляет программисту механизм так называемых "продолжений" (continuations). Этот механизм реализован для задач, т.е., для классов Task и Future<T>, и суть его заключается в том, что с каждой задачей можно связать некоторую другую задачу - продолжение первой задачи. Эта вторая задача будет выполнена после завершения работы первой - основной, задачи.

Библиотека PFX версии June 2008 CTP не поддерживает механизм продолжения для множества задач, когда некоторая задача-продолжение запускается после завершения работы либо всех, либо одной из задач некоторого множества. Однако, такой "множественный" механизм продолжения можно реализовать на основе "единичных" продолжений, класса Future<T> и класса CountDownEvent:

static Task ContinueWhenAll( 
    Action<Task[]> continuation, params Task[] tasks) 
{ 
    var starter = Future<bool>.Create(); 
    var task = starter.ContinueWith(o => continuation(tasks)); 
    CountdownEvent ce = new CountdownEvent(tasks.Length); 
    Action<Task> whenComplete = delegate { 
        if (ce.Decrement()) starter.Value = true; 
    }; 
    foreach (var t in tasks) 
        t.ContinueWith(whenComplete, TaskContinuationKind.OnAny, 
            TaskCreationOptions.None, true); 
    return task; 
}

Суть решения состоит в использовании объекта starter класса Future<T>, который становится готовым (т.е., его свойство Value приобретает значение), когда оканчивают работу все задачи заданного множества задач tasks. Отслеживание окончания работы множества задач происходит с помощью счетчика ce класса CountdownEvent. Уменьшение счетчика на единицу производится каждой задачей из исходного множества задач, а точнее, с помощью задачи-продолжения whenComplete, которая зарегистрирована как задача-продолжение для каждой задачи исходного множества.

Вызывая метод ContinueWhenAll, программист в качестве одного из его параметров указывает список задач, по завершении которых необходимо запустить делегат continuation. В механизме продолжений, реализованном в TPL, делегату-продолжению в качестве параметра передается задача, работу которой он продолжает. Делегату-продолжению в методе ContinueWhenAll, в отличие от стандартного механизма, передается целый массив задач, работу которых этот делегат продолжает.

Ниже приведен пример использования метода ContinueWhenAll:

Task t1 = ..., t2 = ...; 
Task t3 = ContinueWhenAll( 
    delegate { Console.WriteLine("t1 and t2 finished"); }, t1, t2);

Реализация метода ContinueWhenAny, который позволяет запустить исполнение продолжения, когда хотя бы одна задача из множества задач завершилась, похожа на реализацию метода ContinueWhenAll и показана ниже:

static Task ContinueWhenAny( 
    Action<Task> continuation, params Task[] tasks) 
{ 
    WriteOnce<Task> theCompletedTask = new WriteOnce<Task>(); 
    var starter = Future<bool>.Create(); 
    var task = starter.ContinueWith(o => 
        continuation(theCompletedTask.Value)); 
    Action<Task> whenComplete = t => { 
        if (theCompletedTask.TrySetValue(t)) starter.Value = true; 
    }; 
    foreach (var t in tasks) 
         t.ContinueWith(whenComplete, TaskContinuationKind.OnAny, 
            TaskCreationOptions.None, true); 
    return task; 
}

Отметим особенности реализации метода ContinueWhenAll. Для того чтобы отследить момент завершения исполнения какой-либо задачи из множества задач tasks используется переменная с однократным присваиванием WriteOnce< >. Завершившаяся задача пробует присвоить этой переменной ссылку на себя и, если данная задача завершила свое исполнение первой, то метод TrySetValue вернет True и будет запущено исполнение продолжения, иначе метод TrySetValue вернет False, что будет означать, что данная задача завершилась не первой. Остальной код метода ContinueWhenAny аналогичен реализации метода ContinueWhenAll.

9.2 Асинхронное выполнение последовательности задач

Иногда, в некоторых приложениях требуется выполнить друг за другом последовательность задач одновременно (асинхронно) с основным потоком. Другими словами, мы хотели бы создать некоторую перечислимую коллекцию (массив или список) задач и отдать их на выполнение другому потоку. Такой механизм можно реализовать, в частности, с помощью конструкций продолжение ContinueWith:

static void RunAsync(IEnumerable<Task> iterator) 
{ 
    var enumerator = iterator.GetEnumerator(); 
    Action a = null; 
    a = delegate 
    { 
        if (enumerator.MoveNext()) 
        { 
            enumerator.Current.ContinueWith(delegate { a(); }); 
        } 
        else enumerator.Dispose(); 
    }; 
    a(); 
}

Метод RunAsync работает следующим образом: методу RunAsync в качестве аргумента передается перечисляемая коллекция задач; затем метод получает итератор данной коллекции и создает делегат, который будет эту коллекцию перебирать и исполнять содержащиеся в ней задачи. После этого, данный делегат запускается на исполнение. Исполнение делегата заключается в выполнении текущей задачи из исходного списка с регистрацией в качестве продолжения еще одного исполнения этого делегата.

9.3 Ожидание завершения множества задач

В "лекции 5" было описано несколько способов ожидания завершения исполнения одной или нескольких задач с помощью методов Task.Wait и Task.WaitAll.

Предположим, что нам необходимо параллельно обработать некоторую совокупность данных, запуская для обработки одного элемента совокупности отдельную задачу, и затем дождаться завершения работы всех задач. В этом случае, соответствующий фрагмент кода может выглядеть так:

IEnumerable<Data> data = ...; 
List<Task> tasks = new List<Task>(); 
foreach(var item in data) tasks.Add(Task.Create(delegate { Process(item); }); 
Task.WaitAll(tasks.ToArray());

Основным недостатком этого решения является сохранение ссылок на все порожденные задачи с вписке tasks. При большом размере исходной совокупности данных, и, следовательно, большом количестве запущенных задач, сборщик мусора не сможет освободить ресурсы, связанные с уже завершившимися задачами, поскольку ссылки на них собраны в массив tasks, который используется в операторе ожидания WaitAll в качестве аргумента.

На самом деле, в этой ситуации достаточно хранить только значение счетчика запущенных задач. В этом случае, условие "дождаться завершения всех созданных задач" будет эквивалентно условию обнуления этого счетчика. Именно эта идея реализована в классе TaskWaiter, код которого приведен ниже:

public class TaskWaiter 
{ 
    public CountdownEvent _ce = new CountdownEvent(1); 
    private bool _doneAdding = false; 

    public void Add(Task t) 
    { 
        if (t == null) throw new ArgumentNullException("t"); 
        _ce.Increment(); 
        t.ContinueWith(ct => _ce.Decrement()); 
    } 

    public void Wait() 
    { 
        if (!_doneAdding) { _doneAdding = true; _ce.Decrement(); }  
        _ce.Wait(); 
    } 
}

При вызове метода TaskWaiter.Add происходит увеличение счетчика запущенных задач и установка уменьшения счетчика при завершении добавляемой задачи. Первое реализовано на базе потокобезопасного счетчика CountdownEvent, второе - на базе механизма продолжения задач (см. 9.1 и 9.2).

При вызове метода Wait происходит блокирование вызывающего потока. Поток будет заблокирован до тех пор, пока счетчик _ce не примет нулевого значения или, что эквивалентно, до тех пор, пока все запущенные задачи не завершатся.

Пример использования данного класса приведен ниже:

IEnumerable<Data> data = ...; 
TaskWaiter tasks = new TaskWaiter(); 
foreach(var item in data) tasks.Add(Task.Create(delegate { Process(item); }); 
tasks.Wait();

Обратите внимание, что ссылки на запущенные задачи более не сохраняются, что позволяет упростить код и повысить эффективность работы сборщика мусора.

9.4 Реализация конструкции ParallelWhileNotEmpty

Аналогично другим наборам инструментов, библиотека PFX доставляет программисту возможность на основе базовых конструкций, включенных в PFX, строить собственные специальные конструкции (шаблоны) для решения тех или иных задач. Ниже будут показаны способы построения одной из таких конструкций - ParallelWhileNotEmpty. С помощью этой конструкции можно обрабатывать элементы данных из некоторого множества, причем

  • обработка каждого элемента происходит в рамках отдельной параллельной задачи;
  • множество элементов данных может пополниться при такой обработке.

Например, используя эту конструкцию можно запрограммировать обработку дерева, где при обработке отдельного узла в множество необработанных узлов добавляются узлы-потомки данного узла:

ParallelWhileNotEmpty(treeRoots, (node, adder) => 
{ 
    foreach(var child in node.Children) adder(child);  
    Process(node); 
});

Естественно, что существует несколько способов реализации такого шаблона. Рассмотрим некоторые из них.

Во-первых, одно из решений может быть построено на основе отношения "родитель-потомок" между TPL-задачами, которое гарантирует, в частности, завершение работы родительской задачи только после завершения работы всех задач-потомков (если, конечно, они не "отсоединены" от родительской задачи посредством TaskCreationOptions.Detached ). (Вспомнить эти механизмы можно еще раз обратившись к "лекции 5" и разделу 3 данной лекции):

public static void ParallelWhileNotEmpty<T>( 
    IEnumerable<T> initialValues, Action<T, Action<T>> body) 
{ 
    Action<T> addMethod = null; 
    addMethod = v => Task.Create(delegate { body(v, addMethod); }); 
    Parallel.ForEach(initialValues, addMethod); 
}

Ключевым элементом данной реализации является делегат addMethod, который будет создавать (TPL-)задачу, исполняющую функцию обработки body. Аргументами функции body является сам элемент, который нужно обработать (параметр v ), а также делегат, который будет вызван для добавления дополнительных элементов для обработки (например, потомков узла v ). Этим делегатом является сам addMethod, т.е., мы воспользовались рекурсивным вызовом делегатов. Теперь просто остается вызвать метод Parallel.ForEach, передавая ему в качестве аргументов множество элементов, которое нужно обработать, и соответствующий делегат для обработки каждого из этих элементов. Как отмечалось выше, метод Parallel.ForEach не завершит свою работу пока все задачи (и родительская, и дочерние), созданные при вызове делегата addMethod, не будут завершены. Данное решение имеет тот существенный недостаток, что любая задача, у которой существуют потомки, не будет завершена и убрана сборщиком мусора до тех пор, пока не завершат работу ее потомки. Это может привести к большому перерасходу памяти, особенно при работе со структурами с большой степенью вложенности. Устранить этот недостаток можно, воспользовавшись классом TaskWaiter из раздела 3 данной лекции:

public static void ParallelWhileNotEmpty<T>( 
    IEnumerable<T> initialValues, Action<T, Action<T>> body) 
{ 
    TaskWaiter tasks = new TaskWaiter(); 
    Action<T> addMethod = null; 
    addMethod = v => 
        tasks.Add(Task.Create(delegate { body(v, addMethod); }, 
            TaskCreationOptions.Detached)); 

    Parallel.ForEach(initialValues, addMethod); 

    tasks.Wait(); 
}

Из реализации класса TaskWaiter видно (см. раздел 3 данной лекции), что он позволяет разорвать связь "родитель-потомок" между создаваемыми задачами, что ведет к освобождению ресурсов при завершении работы отдельных задач.

Еще одна реализация шаблона ParallelWhileNotEmpty основана на использовании двух списков: в одном из них хранятся элементы, которые обрабатываются на текущей стадии, а в другом накапливаются элементы, которые будут обработаны на следующем шаге. В процессе обработки, эти списки постоянно меняются ролями. Обработка завершается, когда очередной список оказывается пуст.

public static void ParallelWhileNotEmpty<T>( 
    IEnumerable<T> initialValues, Action<T, Action<T>> body) 
{ 
    var lists = new [] { 
     new ConcurrentStack<T>(initialValues), new ConcurrentStack<T>() }; 
    for(int i=0; ; i++) 
    { 
        int fromIndex = i % 2; 
        var from = lists[fromIndex]; 
        var to = lists[fromIndex ^ 1]; 
        if (from.IsEmpty) break; 

        Action<T> addMethod = v => to.Push(v); 
        Parallel.ForEach(from.ToArray(), v => body(v, addMethod)); 
        from.Clear(); 
    } 
}
< Лекция 8 || Лекция 9: 12 || Лекция 10 >
Максим Полищук
Максим Полищук
"...Изучение и анализ примеров.
В и приведены описания и приложены исходные коды параллельных программ..."
Непонятно что такое - "В и приведены описания" и где именно приведены и приложены исходные коды.
Дмитрий Молокоедов
Дмитрий Молокоедов
Россия, Новосибирск, НГПУ, 2009
Паулус Шеетекела
Паулус Шеетекела
Россия, ТГТУ, 2010