Опубликован: 23.01.2013 | Уровень: для всех | Доступ: платный | ВУЗ: Томский политехнический университет
Самостоятельная работа 12:

Пример оптимизации параллельного приложения

< Лекция 13 || Самостоятельная работа 12
Аннотация: В данном практическом занятии будет рассмотрен пример оптимизации параллельного приложения с использованием Currency Visualizer.

  1. Создадим консольное приложение и назовем его, к примеру, "VisualizerConsoleApplication":
  2. Добавим в начало кода выражения: System.Threading и System.Threading.Task, для работы с потоками. В итоге должно получиться следующее:
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading;
    using System.Threading.Tasks;
    
    namespace VisualizerConsoleApplication
    {
        class Program
        {
            static void Main(string[] args)
            {
            }
        }
    }
  3. После чего, добавим в класс Program два статичных метода:
    • Метод DoSomething() в котором будет вызываться метод Thread.SpinWait();
    • Метод ParallelLoop(), в котором перебираются числа от 0 до 1000 (созданные с помощью метода Enumerable.Range()) c использованием цикла foreach. В цикле foreach вызовем метод DoSomething().

    В итоге должно получиться следующее:

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading;
    using System.Threading.Tasks;
    namespace VisualizerConsoleApplication
    {
        class Program
        {
            static void DoSomething()
            {
                Thread.SpinWait(int.MaxValue / 10);
            }
    
            static void ParallelLoop()
            {
                var numbers = Enumerable.Range(0, 1000);
                foreach (var number in numbers)
                {
                    DoSomething();
                }
            }
    
            static void Main(string[] args)
            {
            }
        }
    }
  4. Теперь необходимо вызвать из метода Main() - метод ParallelLoop(), в отдельном потоке. Код будет выглядеть следующим образом:
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading;
    using System.Threading.Tasks;
    namespace VisualizerConsoleApplication
    {
        class Program
        {
            static void DoSomething()
            {
                Thread.SpinWait(int.MaxValue / 10);
            }
            static void ParallelLoop()
            {
                var numbers = Enumerable.Range(0, 1000);
                foreach (var number in numbers)
                {
                    DoSomething();
                }
            }
            static void Main(string[] args)
            {
                new Thread(new ThreadStart(ParallelLoop)).Start();
                Console.ReadLine();
            }
        }
    }
  5. Далее, необходимо запустить Currency Visualizer. Для этого в меню Debug выбрать пункт "Start Performance Analysis" или можно использовать сочетание клавиш Alt+F2 (Для использования Currency Visualizer, VIsual Studio 2010 должна быть запущена с правами администратора)
  6. Запустится окно мастера. В нем необходимо выбрать пункт Concurrency и выбрать в нем два пункта "Collect resource contention data" и "Visualize the behavior of a multithreaded application" и для продолжения работы мастера кнопку Next:
  7. Далее, выбираем проекты для анализа на производительность (если их несколько) и жмем кнопку "Next":
  8. На последнем шаге мастера, выберите пункт "Launch profiling after the wizard finish", после чего нажмите кнопку "Finish":
  9. После чего запустится окно программы (в нашем случае это консоль) и процесс профилирования. Остановите процесс профилирования приложения с помощью кнопки "Stop Profiling" через 10-15 секунд:
  10. После завершения процесса анализа производительности, отобразится окно с различными отчетами:
  11. Выберем из списка "Cores". Отобразиться график, который показывает дисбаланс нагрузки на логические ядра процессора:
  12. Для того что бы сбалансировать нагрузку на ядра процессора модифицируем метод ParallelLoop() - заменим цикл foreach на Parallel.ForEach:
    static void ParallelLoop()
         {
             var numbers = Enumerable.Range(0, 1000);
             Parallel.ForEach(numbers, (number) =>
             {
                 DoSomething();
             });
    
         }
  13. Повторно запустим профилирование. На диаграмме видно, что теперь каждое ядро процессора имеет сбалансированную нагрузку, кроме того каждое ядро выполняет различные потоки:
  14. Выберите вкладку "CPU Utilization". На данном графике отображается использование процессора с работающим приложением (в нашем случае приложение использует 54%):
  15. Теперь, добавим в приложение два идентичных потока и один объект sync (который будет использоваться для синхронизации lock):
    static void Main(string[] args)
            {
                new Thread(new ThreadStart(ParallelLoop)).Start();
    
                object sync = new object();
                new Thread(new ThreadStart(() =>
                {
                    lock (sync)
                    {
                        Thread.Sleep(2000);
                    }
    
                })).Start();
    
                new Thread(new ThreadStart(() => 
                {
                    lock (sync)
                    {
                        Thread.Sleep(2000);
                    }
    
                })).Start();
                           
                Console.ReadLine();
            }

    Примечание: ключевое слово lock не позволит одному потоку войти в раздел кода в тот момент, когда в нем находится другой поток.

  16. Запустим повторно профилирование. После профилирования перейдем во вкладку "Threads":

    На диаграмме можно видеть два зависимых потока, один из потоков запускается с сегмента синхронизации (Thread.Sleep(2000)).

    Листинг кода программы:

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading;
    using System.Threading.Tasks;
    namespace VisualizerConsoleApplication
    {
        class Program
        {
            static void DoSomething()
            {
                Thread.SpinWait(int.MaxValue / 10);
            }
    
            static void ParallelLoop()
            {
                var numbers = Enumerable.Range(0, 1000);
                Parallel.ForEach(numbers, (number) =>
                {
                    DoSomething();
                });
            }
            static void Main(string[] args)
            {
                new Thread(new ThreadStart(ParallelLoop)).Start();
                object sync = new object();
                new Thread(new ThreadStart(() =>
                {
                    lock (sync)
                    {
                        Thread.Sleep(2000);
                    }
    
                })).Start();
    
                new Thread(new ThreadStart(() =>
                {
                    lock (sync)
                    {
                        Thread.Sleep(2000);
                    }
    
          })).Start();
    
                Console.ReadLine();
    
            }
        }
    }
< Лекция 13 || Самостоятельная работа 12
Владимир Каширин
Владимир Каширин

Вопрос по Курсу: "Параллельное программирование с использованием MS VisualStudia 2010".

При компиляции Самостоятельного задания (одновременная отрисовка прямоугольников, эллипсов и выдача в текст-бокс случайного числа) среда предупреждает: suspend - устаревшая команда; примените monitor, mutex и т.п.

Создаётся впечатление, что Задание создано в более поздней среде, чем VS 2010.

Александр Гаврилов
Александр Гаврилов
Россия
Роман Дмитриев
Роман Дмитриев
Россия, Москва