Рейтинговые книги
Читем онлайн Язык программирования C#9 и платформа .NET5 - Эндрю Троелсен

Шрифт:

-
+

Интервал:

-
+

Закладка:

Сделать
1 ... 188 189 190 191 192 193 194 195 196 ... 407
            SearchOption.AllDirectories);

  string outputDirectory = @".ModifiedPictures";

  Directory.CreateDirectory(outputDirectory);

  try

  {

    // Обработать данные изображения в параллельном режиме!

    Parallel.ForEach(files, parOpts, currentFile =>

    {

      parOpts

         .CancellationToken.ThrowIfCancellationRequested();

      string filename = Path.GetFileName(currentFile);

      Dispatcher?.Invoke(() =>

      {

        this.Title =

          $"Processing {filename}

             on thread {Thread.CurrentThread.ManagedThreadId}";

      });

      using (Bitmap bitmap = new Bitmap(currentFile))

      {

        bitmap.RotateFlip(RotateFlipType.Rotate180FlipNone);

        bitmap.Save(Path.Combine(outputDirectory, filename));

      }

    });

    Dispatcher?.Invoke(()=>this.Title = "Done!");

  }

  catch (OperationCanceledException ex)

  {

    Dispatcher?.Invoke(()=>this.Title = ex.Message);

  }

}

Обратите внимание, что в начале метода конфигурируется объект ParallelOptions с установкой его свойства CancellationToken для применения маркера CancellationTokenSource. Кроме того, этот объект ParallelOptions передается во втором параметре методу Parallel.ForEach().

Внутри логики цикла осуществляется вызов ThrowIfCancellationRequested() на маркере отмены, гарантируя тем самым, что если пользователь щелкнет на кнопке Cancel, то все потоки будут остановлены ив качестве уведомления сгенерируется исключение времени выполнения. Перехватив исключение OperationCanceledException, можно добавить в текст главного окна сообщение об ошибке.

Обеспечение параллелизма задач с помощью класса Parallel

В дополнение к обеспечению параллелизма данных библиотека TPL также может использоваться для запуска любого количества асинхронных задач с помощью метода Parallel.Invoke(). Такой подход немного проще, чем применение делегатов или типов из пространства имен System.Threading, но если нужна более высокая степень контроля над выполняемыми задачами, тогда следует отказаться от использования Parallel.Invoke() и напрямую работать с классом Task, как делалось в предыдущем примере.

Чтобы взглянуть на параллелизм задач в действии, создайте новый проект консольного приложения по имени MyEBookReader и импортируйте в начале файла Program.cs пространства имен System.Threading, System.Text, System.Threading.Tasks, System.Linq и System.Net (пример является модификацией полезного примера из документации по .NET Core). Здесь мы будем извлекать публично доступную электронную книгу из сайта проекта Гутенберга (www.gutenberg.org) и затем параллельно выполнять набор длительных задач. Книга загружается в методе GetBook(), показанном ниже:

using System;

using System.Linq;

using System.Threading;

using System.Threading.Tasks;

using System.Net;

using System.Text;

string _theEBook = "";

GetBook();

Console.WriteLine("Downloading book...");

Console.ReadLine();

void GetBook()

{

  WebClient wc = new WebClient();

  wc.DownloadStringCompleted += (s, eArgs) =>

  {

    _theEBook = eArgs.Result;

    Console.WriteLine("Download complete.");

    GetStats();

  };

  // Загрузить электронную книгу Чарльза Диккенса "A Tale of Two Cities".

  // Может понадобиться двукратное выполнение этого кода, если ранее вы

  // не посещали данный сайт, поскольку при первом его посещении появляется

  // окно с сообщением, предотвращающее нормальное выполнение кода.

  wc.DownloadStringAsync(new Uri("http://www.gutenberg.org/

  files/98/98-8.txt"));

}

Класс WebClient определен в пространстве имен System.Net. Он предоставляет несколько методов для отправки и получения данных от ресурса, идентифицируемого посредством URL. В свою очередь многие из них имеют асинхронные версии, такие как метод DownloadStringAsync(), который автоматически порождает новый поток из пула потоков .NET Core Runtime. Когда объект WebClient завершает получение данных, он инициирует событие DownloadStringCompleted, которое обрабатывается с применением лямбда-выражения С#. Если вызвать синхронную версию этого метода (DownloadString()), то сообщение Downloading book... не появится до тех пор, пока загрузка не завершится.

Далее реализуйте метод GetStats() для извлечения индивидуальных слов, содержащихся в переменной theEBook, и передачи строкового массива на обработку нескольким вспомогательным методам:

void GetStats()

{

  // Получить слова из электронной книги.

  string[] words = _theEBook.Split(new char[]

    { ' ', 'u000A', ',', '.', ';', ':', '-', '?', '/' },

    StringSplitOptions.RemoveEmptyEntries);

  // Найти 10 наиболее часто встречающихся слов.

  string[] tenMostCommon = FindTenMostCommon(words);

  // Получить самое длинное слово.

  string longestWord = FindLongestWord(words);

  // Когда все задачи завершены, построить строку, показывающую

  // все статистические данные в окне сообщений.

  StringBuilder bookStats =

      new StringBuilder("Ten Most Common Words are:n");

  foreach (string s in tenMostCommon)

  {

    bookStats.AppendLine(s);

  }

  bookStats.AppendFormat("Longest word is: {0}", longestWord);

                       // Самое длинное слово

  bookStats.AppendLine();

  Console.WriteLine(bookStats.ToString(), "Book info");

                                        // Информация о книге

}

Метод FindTenMostCommon() использует запрос LINQ для получения списка объектов string, которые наиболее часто встречаются в массиве string, а метод FindLongestWord() находит самое длинное слово:

string[] FindTenMostCommon(string[] words)

{

    var frequencyOrder = from word in words

                         where word.Length > 6

                         group word by word into g

                         orderby g.Count() descending

                         select g.Key;

    string[] commonWords = (frequencyOrder.Take(10)).ToArray();

    return commonWords;

}

string FindLongestWord(string[] words)

{

    return (from w in words orderby w.Length descending select w)

        .FirstOrDefault();

}

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

Downloading book...

Download complete.

Ten Most Common Words are:

Defarge

himself

Manette

through

nothing

business

another

looking

prisoner

Cruncher

Longest word is: undistinguishable

Помочь удостовериться в том, что приложение задействует все доступные процессоры машины, может параллельный вызов методов FindTenMostCommon() и FindLongestWord(). Для этого модифицируйте метод GetStats():

void GetStats()

{

  // Получить слова из электронной книги.

  string[] words = _theEBook.Split(

    new char[] { ' ', 'u000A', ',', '.', ';', ':', '-', '?', '/' },

    StringSplitOptions.RemoveEmptyEntries);

  string[] tenMostCommon = null;

  string longestWord = string.Empty;

  Parallel.Invoke(

    () =>

    {

      // Найти 10 наиболее

1 ... 188 189 190 191 192 193 194 195 196 ... 407
На этой странице вы можете бесплатно читать книгу Язык программирования C#9 и платформа .NET5 - Эндрю Троелсен бесплатно.

Оставить комментарий