Книга: C# 4.0: полное руководство

Класс Parallel

Класс Parallel

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

Класс Parallel является статическим, и в нем определены методы For(), For Each() и Invoke(). У каждого из этих методов имеются различные формы. В частности, метод For() выполняет распараллеливаемый цикл for, а метод ForEach() — распараллеливаемый цикл foreach, и оба метода поддерживают параллелизм данных. А метод Invoke() поддерживает параллельное выполнение двух методов или больше. Как станет ясно дальше, эти методы дают преимущество реализации на практике распространенных методик параллельного программирования, не прибегая к управлению задачами или потоками явным образом. В последующих разделах каждый из этих методов будет рассмотрен более подробно.

Распараллеливание задач методом Invoke()

Метод Invoke(), определенный в классе Parallel, позволяет выполнять один или несколько методов, указываемых в виде его аргументов. Он также масштабирует исполнение кода, используя доступные процессоры, если имеется такая возможность. Ниже приведена простейшая форма его объявления.

public static void Invoke(params Action[] actions)

Выполняемые методы должны быть совместимы с описанным ранее делегатом Action. Напомним, что делегат Action объявляется следующим образом.

public delegate void Action()

Следовательно, каждый метод, передаваемый методу Invoke() в качестве аргумента, не должен ни принимать параметров, ни возвращать значение. Благодаря тому что параметр actions данного метода относится к типу params, выполняемые методы могут быть указаны в виде переменного списка аргументов. Для этой цели можно также воспользоваться массивом объектов типа Action, но зачастую оказывается проще указать список аргументов.

Метод Invoke() сначала инициирует выполнение, а затем ожидает завершения всех передаваемых ему методов. Это, в частности, избавляет от необходимости (да и не позволяет) вызывать метод Wait(). Все функции параллельного выполнения метод Wait() берет на себя. И хотя это не гарантирует, что методы будут действительно выполняться параллельно, тем не менее, именно такое их выполнение предполагается, если система поддерживает несколько процессоров. Кроме того, отсутствует возможность указать порядок выполнения методов от первого и до последнего, и этот порядок не может быть таким же, как и в списке аргументов.

В приведенном ниже примере программы демонстрируется применение метода Invoke() на практике. В этой программе два метода MyMeth() и MyMeth2() выполняются параллельно посредством вызова метода Invoke(). Обратите внимание на простоту организации данного процесса.

// Применить метод Parallel.Invoke() для параллельного выполнения двух методов.
using System;
using System.Threading;
using System.Threading.Tasks;
class DemoParallel {
  // Метод, исполняемый как задача,
  static void MyMeth() {
    Console.WriteLine("MyMeth запущен");
    for (int count = 0; count < 5; count++) {
      Thread.Sleep(500);
      Console.WriteLine("В методе MyMeth подсчет равен " + count );
    }
    Console.WriteLine("MyMeth завершен");
  }
  // Метод, исполняемый как задача,
  static void MyMeth2() {
    Console.WriteLine("MyMeth2 запущен");
    for(int count = 0; count < 5; count++) {
      Thread.Sleep(500);
      Console.WriteLine("В методе MyMeth2, подсчет равен " + count );
    }
    Console.WriteLine("MyMeth2 завершен");
  }
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    // Выполнить параллельно два именованных метода.
    Parallel.Invoke(MyMeth, MyMeth2);
    Console.WriteLine("Основной поток завершен.");
  }
}

Выполнение этой программы может привести к следующему результату.

Основной поток запущен.
MyMeth запущен
MyMeth2 запущен
В методе MyMeth подсчет равен 0
В методе MyMeth2, подсчет равен 0
В методе MyMeth подсчет равен 1
В методе MyMeth2, подсчет равен 1
В методе MyMeth подсчет равен 2
В методе MyMeth2, подсчет равен 2
В методе MyMeth подсчет равен 3
В методе MyMeth2, подсчет равен 3
В методе MyMeth подсчет равен 4
MyMeth завершен
В методе MyMeth2, подсчет равен 4
MyMeth2 завершен
Основной поток завершен.

В данном примере особое внимание обращает на себя следующее обстоятельство: выполнение метода Main() приостанавливается до тех пор, пока не произойдет возврат из метода Invoke(). Следовательно, метод Main(), в отличие от методов MyMeth() и MyMeth2(), не выполняется параллельно. Поэтому применять метод Invoke() показанным здесь способом нельзя в том случае, если требуется, чтобы исполнение вызывающего потока продолжалось.

В приведенном выше примере использовались именованные методы, но для вызова метода Invoke() это условие не является обязательным. Ниже приведен переделанный вариант той же самой программы, где в качестве аргументов в вызове метода Invoke() применяются лямбда-выражения.

// Применить метод Parallel.Invoke()
//для параллельного выполнения двух методов.
// В этой версии программы применяются лямбда-выражения.
using System;
using System.Threading;
using System.Threading.Tasks;
class DemoParallel {
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    // Выполнить два анонимных метода, указываемых в лямбда-выражениях.
    Parallel.Invoke(() => {
      Console.WriteLine("Выражение #1 запущено");
      for(int count = 0; count < 5; count++) {
        Thread.Sleep(500);
        Console.WriteLine("В выражении #1 подсчет равен " + count );
      }
      Console.WriteLine("Выражение #1 завершено");
    },
    () => {
      Console.WriteLine("Выражение #2 запущено");
      for (int count = 0; count < 5; count++) {
        Thread.Sleep(500);
        Console.WriteLine("В выражении #2 подсчет равен " + count );
      }
      Console.WriteLine("Выражение #1 завершено");
    } );
    Console.WriteLine("Основной поток завершен.");
  }
}

Эта программа дает результат, похожий на результат выполнения предыдущей программы.

Применение метода For()

В TPL параллелизм данных поддерживается, в частности, с помощью метода For(), определенного в классе Parallel. Этот метод существует в нескольких формах. Его рассмотрение мы начнем с самой простой формы, приведенной ниже:

public static ParallelLoopResult
                For(int fromlnclusive, int toExclusive, Action<int> body)

где fromlnclusive обозначает начальное значение того, что соответствует переменной управления циклом; оно называется также итерационным, или индексным, значением; a toExclusive — значение, на единицу больше конечного. На каждом шаге цикла переменная управления циклом увеличивается на единицу. Следовательно, цикл постепенно продвигается от начального значения fromlnclusive к конечному значению toExclusive минус единица. Циклически выполняемый код указывается методом, передаваемым через параметр body. Этот метод должен быть совместим с делегатом Action<int>, объявляемым следующим образом.

public delegate void Action<in T>(T obj)

Для метода For() обобщенный параметр Т должен быть, конечно, типа int. Значение, передаваемое через параметр obj, будет следующим значением переменной управления циклом. А метод, передаваемый через параметр body, может быть именованным или анонимным. Метод For() возвращает экземпляр объекта типа ParallelLoopResult, описывающий состояние завершения цикла. Для простых циклов этим значением можно пренебречь. (Более подробно это значение будет рассмотрено несколько ниже.)

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

В приведенном ниже примере программы демонстрируется применение метода For() на практике. В начале этой программы создается массив data, состоящий из 1000000000 целых значений. Затем вызывается метод For(), которому в качестве "тела" цикла передается метод MyTransfогm(). Этот метод состоит из ряда операторов, выполняющих произвольные преобразования в массиве data. Его назначение — сымитировать конкретную операцию. Как будет подробнее пояснено несколько ниже, выполняемая операция должна быть нетривиальной, чтобы параллелизм данных принес какой-то положительный эффект. В противном случае последовательное выполнение цикла может завершиться быстрее.

// Применить метод Parallel.For() для организации параллельно
// выполняемого цикла обработки данных.
using System;
using System.Threading.Tasks;
class DemoParallelFor {
  static int[] data;
  // Метод, служащий в качестве тела параллельно выполняемого цикла.
  // Операторы этого цикла просто расходуют время ЦП для целей демонстрации,
  static void MyTransform(int i) {
    data[i] = data[i] / 10;
    if(data[i] < 10000) data[i] = 0;
    if(data[i] > 10000 & data[i] < 20000) data[i] = 100;
    if(data[i] > 20000 & data[i] < 30000) data[i] = 200;
    if(data[i] > 30000) data[i] = 300;
  }
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    data = new int[100000000];
    // Инициализировать данные в обычном цикле for.
    for (int i=0; i < data.Length; i++) data[i] = i;
    // Распараллелить цикл методом For().
    Parallel.For(0, data.Length, MyTransform);
    Console.WriteLine("Основной поток завершен.");
  }
}

Эта программа состоит из двух циклов. В первом, стандартном, цикле for инициализируется массив data. А во втором цикле, выполняемом параллельно методом For(), над каждым элементом массива data производится преобразование. Как упоминалось выше, это преобразование носит произвольный характер и выбрано лишь для целей демонстрации. Метод For() автоматически разбивает вызовы метода MyTransform() на части для параллельной обработки отдельных порций данных, хранящихся в массиве. Следовательно, если запустить данную программу на компьютере с двумя доступными процессорами или больше, то цикл преобразования данных в массиве может быть выполнен методом For() параллельно.

Следует, однако, иметь в виду, что далеко не все циклы могут выполняться эффективно, когда они распараллеливаются. Как правило, мелкие циклы, а также циклы, состоящие из очень простых операций, выполняются быстрее последовательным способом, чем параллельным. Именно поэтому цикл for инициализации массива данных не распараллеливается методом For() в рассматриваемой здесь программе. Распараллеливание мелких и очень простых циклов может оказаться неэффективным потому, что время, требующееся для организации параллельных задач, а также время, расходуемое на переключение контекста, превышает время, экономящееся благодаря параллелизму. В подтверждение этого факта в приведеннрм ниже примере программы создаются последовательный и параллельный варианты цикла for, а для сравнения на экран выводится время выполнения каждого из них.

// Продемонстрировать отличия во времени последовательного
//и параллельного выполнения цикла for.
using System;
using System.Threading.Tasks;
using System.Diagnostics;
class DemoParallelFor {
  static int[] data;
  // Метод, служащий в качестве тела параллельно выполняемого цикла.
  // Операторы этого цикла просто расходуют время ЦП для целей демонстрации,
  static void MyTransform(int i) {
    data[i] = data[i] / 10;
    if(data[i] < 1000) data[i] = 0;
    if(data[i] > 1000 & data[i] < 2000) data[i] = 100;
    if(data[i] > 2000 & data[i] < 3000) data[i] = 200;
    if(data[i] > 3000) data[i] = 300;
  }
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    // Create экземпляр объекта типа Stopwatch
    // для хранения времени выполнения цикла.
    Stopwatch sw = new Stopwatch();
    data = new int[100000000];
    // Инициализировать данные,
    sw.Start();
    // Параллельный вариант инициализации массива в цикле.
    Parallel.For(0, data.Length, (i) => data[i] = i );
    sw.Stop();
    Console.WriteLine("Параллельно выполняемый цикл инициализации: " +
          "{0} секунд", sw.Elapsed.TotalSeconds);
    sw.Reset();
    sw.Start();
    // Последовательный вариант инициализации массива в цикле,
    for(int i=0; i < data.Length; i++) data[i] = i;
    sw.Stop();
    Console.WriteLine("Последовательно выполняемый цикл инициализации: " +
          "{0} секунд", sw.Elapsed.TotalSeconds);
    Console.WriteLine();
    // Выполнить преобразования,
    sw.Start();
    // Параллельный вариант преобразования данных в цикле.
    Parallel.For(0, data.Length, MyTransform);
    sw.Stop();
    Console.WriteLine("Параллельно выполняемый цикл преобразования: " +
          "{0} секунд", sw.Elapsed.TotalSeconds);
    sw.Reset();
    sw.Start();
    // Последовательный вариант преобразования данных в цикле,
    for(int i=0; i < data.Length; i++) MyTransform(i);
    sw.Stop();
    Console.WriteLine("Последовательно выполняемый цикл преобразования: " +
          "{0} секунд", sw.Elapsed.TotalSeconds);
    Console.WriteLine("Основной поток завершен.");
  }
}

При выполнении этой программы на двухъядерном компьютере получается следующий результат.

Основной поток запущен.
Параллельно выполняемый цикл инициализации: 1.0537757 секунд
Последовательно выполняемый цикл инициализации: 0.3457628 секунд
Параллельно выполняемый цикл преобразования: 4.2246675 секунд
Последовательно выполняемый цикл преобразования: 5.3849959 секунд
Основной поток завершен.

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

----------------------------------

ПРИМЕЧАНИЕ

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

----------------------------------

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

Parallel.For(0, data.Length, (i) => data[i] = i );

Здесь "тело" цикла указывается в лямбда-выражении. (Напомним, что в лямбда-выражении создается анонимный метод.) Следовательно, для параллельного выполнения методом For() совсем не обязательно указывать именованный метод.

И во-вторых, обратите внимание на применение класса Stopwatch для вычисления времени выполнения цикла. Этот класс находится в пространстве имен System.Diagnostics. Для того чтобы воспользоваться им, достаточно создать экземпляр его объекта, а затем вызвать метод Start(), начинающий отчет времени, и далее — метод Stop(), завершающий отсчет времени. А с помощью метода Reset() отсчет времени сбрасывается в исходное состояние. Продолжительность выполнения можно получить различными способами. В рассматриваемой здесь программе для этой цели использовано свойство Elapsed, возвращающее объект типа TimeSpan. С помощью этого объекта и свойства TotalSeconds время отображается в секундах, включая и доли секунды. Как показывает пример рассматриваемой здесь программы, класс Stopwatch оказывается весьма полезным при разработке параллельно исполняемого кода.

Как упоминалось выше, метод For() возвращает экземпляр объекта типа ParallelLoopResult. Это структура, в которой определяются два следующих свойства.

public bool IsCompleted { get; }
public Nullable<long> LowestBreaklteration { get; }

Свойство IsCompleted будет иметь логическое значение true, если выполнены все шаги цикла. Иными словами, при нормальном завершении цикла это свойство будет содержать логическое значение true. Если же выполнение цикла прервется раньше времени, то данное свойство будет содержать логическое значение false. Свойство LowestBreaklteration будет содержать наименьшее значение переменной управления циклом, если цикл прервется раньше времени вызовом метода ParallelLoopState.Break().

Для доступа к объекту типа ParallelLoopState следует использовать форму метода For(), делегат которого принимает в качестве второго параметра текущее состояние цикла. Ниже эта форма метода For() приведена в простейшем виде.

public static ParallelLoopResult For(int fromlnclusive, int toExclusive,
Action<int, ParallelLoopState> body)

В данной форме делегат Action, описывающий тело цикла, определяется следующим образом.

public delegate void Action<in T1, in T2>(T arg1, T2 arg2)

Для метода For() обобщенный параметр T1 должен быть типа int, а обобщенный параметр Т2 — типа ParallelLoopState. Всякий раз, когда делегат Action вызывается, текущее состояние цикла передается в качестве аргумента arg2.

Для преждевременного завершения цикла следует воспользоваться методом Break(), вызываемым для экземпляра объекта типа ParallelLoopState внутри тела цикла, определяемого параметром body. Метод Break() объявляется следующим образом.

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

Прерывание цикла, параллельно выполняемого методом For(), нередко оказывается полезным при поиске данных. Так, если искомое значение найдено, то продолжать выполнение цикла нет никакой надобности. Прерывание цикла может оказаться полезным и в том случае, если во время очередной операции встретились недостоверные данные.

В приведенном ниже примере программы демонстрируется применение метода Break() для прерывания цикла, параллельно выполняемого методом For(). Это вариант предыдущего примера, переработанный таким образом, чтобы метод MyTransform() принимал теперь объект типа ParallelLoopState в качестве своего параметра, а метод Break() вызывался при обнаружении отрицательного значения в массиве данных. Отрицательное значение, по которому прерывается выполнение цикла, вводится в массив data внутри метода Main(). Далее проверяется состояние завершения цикла преобразования данных. Свойство IsCompleted будет содержать логическое значение false, поскольку в массиве data обнаруживается отрицательное значение. При этом на экран выводится номер шага, на котором цикл был прерван. (В этой программе исключены все избыточные циклы, применявшиеся в ее предыдущей версии, а оставлены только самые эффективные из них: последовательно выполняемый цикл инициализации и параллельно выполняемый цикл преобразования.)

// Использовать объекты типа ParallelLoopResult и ParallelLoopState, а также
// метод Break() вместе с методом For() для параллельного выполнения цикла.
using System;
using System.Threading.Tasks;
class DemoParallelForWithLoopResult {
  static int[] data;
  // Метод, служащий в качестве тела параллельно выполняемого цикла.
  //Операторы этого цикла просто расходуют время ЦП для целей демонстрации,
  static void MyTransform(int i, ParallelLoopState pis) {
    // Прервать цикл при обнаружении отрицательного значения,
    if(data[i] < 0) pis.Break();
    data[i] = data[i] / 10;
    if(data[i] < 1000) data[i] = 0;
    if(data[i] > 1000 & data[i] < 2000) data[i] = 100;
    if(data[i] > 2000 & data[i] < 3000) data[i] = 200;
    if(data[i] > 3000) data[i] = 300;
  }
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    data = new int[100000000];
    // Инициализировать данные.
    for(int i=0; i < data.Length; i++) data[i] = i;
    // Поместить отрицательное значение в массив data, data[1000] = -10;
    // Параллельный вариант инициализации массива в цикле.
    ParallelLoopResult loopResult = Parallel.For(0, data.Length, MyTransform);
    // Проверить, завершился ли цикл,
    if(!loopResult.IsCompleted)
      Console.WriteLine("nЦикл завершился преждевременно из-за того, " +
            "что обнаружено отрицательное значение" +
            "на шаге цикла номер " +
            loopResult.LowestBreakIteration + ".n");
    Console.WriteLine("Основной поток завершен.");
  }
}

Выполнение этой программы может привести, например, к следующему результату.

Основной поток запущен.
Цикл завершился преждевременно из-за того, что обнаружено отрицательное значение на шаге цикла номер 1000
Основной поток завершен.

Как следует из приведенного выше результата, цикл преобразования данных преждевременно завершается после 1000 шагов. Дело в том, что метод Break() вызывается внутри метода MyTransform() при обнаружении в массиве данных отрицательного значения.

Помимо двух описанных выше форм метода For() существует и ряд других его форм. В одних из этих форм допускается указывать различные дополнительные параметры, а в других — использовать параметры типа long вместо int для пошагового выполнения цикла. Имеются также формы метода For(), предоставляющие такие дополнительные преимущества, как, например, возможность указывать метод, вызываемый по завершении потока каждого цикла.

И еще одно, последнее замечание: если требуется остановить цикл, параллельно выполняемый методом For(), не обращая особого внимания на любые шаги цикла, которые еще могут быть в нем выполнены, то для этой цели лучше воспользоваться методом Stop(), чем методом Break().

Применение метода ForEach()

Используя метод ForEach(), можно создать распараллеливаемый вариант цикла foreach. Существует несколько форм метода ForEach(). Ниже приведена простейшая форма его объявления:

public static ParallelLoopResult
      ForEach<TSource>(IEnumerable<TSource> source,
               Action<TSource> body)

где source обозначает коллекцию данных, обрабатываемых в цикле, a body — метод, который будет выполняться на каждом шаге цикла. Как пояснялось ранее в этой книге, во всех массивах, коллекциях (описываемых в главе 25) и других источниках данных поддерживается интерфейс IEnumerable<T>. Метод, передаваемый через параметр body, принимает в качестве своего аргумента значение или ссылку на каждый обрабатываемый в цикле элемент массива, но не его индекс. А в итоге возвращаются сведения о состоянии цикла.

Аналогично методу For(), параллельное выполнение цикла методом ForEach() можно остановить, вызвав метод Break() для экземпляра объекта типа ParallelLoopState, передаваемого через параметр body, при условии, что используется приведенная ниже форма метода ForEach().

public static ParallelLoopResult
      ForEach<TSource>(IEnumerable<TSource> source,
          Action<TSource, ParallelLoopState> body)

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

// Использовать объекты типа ParallelLoopResult и ParallelLoopState, а также
// метод Break() вместе с методом ForEach() для параллельного выполнения цикла.
using System;
using System.Threading.Tasks;
class DemoParallelForWithLoopResult {
  static int[] data;
  // Метод, служащий в качестве тела параллельно выполняемого цикла.
  // В данном примере переменной v передается значение элемента массива
  // данных, а не индекс этого элемента.
  static void DisplayData(int v, ParallelLoopState pis) {
    // Прервать цикл при обнаружении отрицательного значения,
    if (v < 0) pis.Break();
    Console.WriteLine("Значение: " + v);
  }
  static void Main() {
    Console.WriteLine("Основной поток запущен.");
    data = new int[100000000];
    // Инициализировать данные.
    for (int i=0; i < data.Length; i++) data[i] = i;
    // Поместить отрицательное значение в массив data,
    data[100000] = -10;
    // Использовать цикл, параллельно выполняемый методом ForEach(),
    // для отображения данных на экране.
    ParallelLoopResult loopResult = Parallel.ForEach(data, DisplayData);
    // Проверить, завершился ли цикл,
    if(!loopResult.IsCompleted)
      Console.WriteLine("nЦикл завершился преждевременно из-за того, " +
             "что обнаружено отрицательное значение" +
             "на шаге цикла номер " +
             loopResult.LowestBreakIteration + ".n");
    Console.WriteLine("Основной поток завершен.");
  }
}

В приведенной выше программе именованный метод применяется в качестве делегата, представляющего "тело" цикла. Но иногда удобнее применять анонимный метод. В качестве примера ниже приведено реализуемое в виде лямбда-выражения "тело" цикла, параллельно выполняемого методом ForEach().

// Использовать цикл, параллельно выполняемый методом ForEach(),
// для отображения данных на экране.
ParallelLoopResult loopResult =
        Parallel.ForEach(data, (v, pis) => {
                Console.WriteLine("Значение: " + v);
                if (v < 0) pis.Break();
         }
         );

Оглавление книги


Генерация: 0.269. Запросов К БД/Cache: 2 / 0
поделиться
Вверх Вниз