Многопоточность в C#: как работать с потоками в C# и как в потоке обращаться к элементам формы

Многопоточность в C#: как работать с потоками в C# и как в потоке обращаться к элементам формы
Наступил день, когда мне понадобилась многопоточность в C#. Я мог бы и не использовать в приложении многопоточность, но тогда в момент работы приложения создаётся впечатление, что приложение зависает в момент, когда приложение ожидает завершения какой-либо операции через .Wait().

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

Как работать с потоками в C#

После изучения данной темы, проб и ошибок, упрощения и написания кода всё оказалось достаточно просто. Сейчас постараюсь простыми словами объяснить, как запустить задачу в отдельном потоке.

Подключаем:

Далее у нас есть некая функция 1 (func1) в которой выполняется некий код и в середине этого кода у нас есть кусок кода, который мы хотим выполнить в отдельном потоке. Для этого нам необходимо написать дополнительную функция 2 (func2), в которую вынести весь код, который мы будем запускать в отдельном потоке. После этого в нужном месте функции 1 нам надо вызвать отдельный поток и продолжить выполнение кода, в то время, когда в отдельном потоке параллельно будет выполняться другой код. Теперь всё вышесказанное мы опишем кодом.

Функция 2:

Функция 1:

Функции привёл в обратном порядке, чтобы далее прокомментировать функцию 1. Сначала у нас выполняется некий «КОД 1» после этого мы запускаем первый поток и сразу продолжаем выполнять «КОД 2», потом доходим до запуска второго потока и его запускаем и сразу переходим к выполнению следующего кода «КОД 3». В итоге у нас два потока выполняют код из функций «func2» и «func3» одновременно с выполнением кода основной функции «func1».

В данном примере у нас после выполнения «КОД 3» происходит выход из функции 1 при этом потоки продолжают свою работу пока не выполнится весь вызываемый ими код. Есть способ, который позволяет при завершении главной функции, которая запустила потоки, сразу завершать и выполнение потоков, для этого добавляем потоку параметр «IsBackground = true»:

Теперь у нас поток будет завершён сразу после выполнения «КОД 2».

Если вы хотите узнать в каком состоянии сейчас находится запущенный вами поток, то вам необходимо получить его статус «potok1.ThreadState». Подробное описание статусов можете просмотреть на сайте Microsoft.

Как в отдельном потоке в C# обращаться к элементам формы

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

Объявляем «MyDelegate»:

Обратите внимание, где он объявляется.

Теперь создадим функцию, в которой будем изменять требуемый нам элемент:

Теперь в коде отдельного потока мы с вами изменим элемент обратившись к функции изменяющий требуемый элемент:

А что делать если нам надо изменить элемент передав в него какое-то значение? Для этого объявляем «MyDelegate» с переменной:

И создаём функцию, в которой будем изменять требуемый нам элемент, с такой же переменной.

Теперь в коде отдельного потока изменим элемент обратившись к функции изменяющий требуемый элемент и передав требуемое нами значение:

Основа заложена, а дальше уже остаётся самостоятельно постигать глубинный смысл потоков 🙂

9 комментариев

  1. Эта статья мне очень помогла т.к. ранее не сталкивался с многопоточностью. Описано все очень доходчиво и просто. Спасибо =)

  2. Дмитрий

    Самая доходчивая статья многопоточности C#.
    Спасибо!

  3. Блин — это ЛУЧШЕЕ объяснение многопоточности и их вызовов в формах.
    Спасибо

  4. Спасибо!

  5. Дмитрий

    Спасибо!

  6. Чувак Спасибо! Реально выручил!

  7. Сам много раз читал в разных источниках дебильные или пространные объяснения принципов многопоточности, спасибо автору, что у него всё в порядке и с пониманием физики процесса и с умением донести инфу до других.

  8. а как сделать так, чтобы форма не морозилась?
    сделал в IzmeniElement цикл <100 с Thread.Sleep(3000);
    вызываю в button1_Click, форма морозится.
    Как это обойти?

    • Если быстро, то как-то так попробуй:

      using System;
      using System.Windows.Forms;
      using System.Threading;
      
      namespace test_01
      {
          public partial class Form1 : Form
          {
              public delegate void MyDelegate();
      
              public Form1()
              {
                  InitializeComponent();
              }
      
              private void button1_Click(object sender, EventArgs e)
              {
                  Thread potok1 = new Thread(func1); // создание отдельного потока
                  potok1.IsBackground = true;
                  potok1.Start(); // запуск потока
              }
      
              public void func1()
              {
                  for (int i = 0; i < 100; i++)
                  {
                      Thread.Sleep(3000);
                  }
                  BeginInvoke(new MyDelegate(IzmeniElement)); // изменяем элемент
              }
      
              public void IzmeniElement()
              {
                  label1.Visible = false; // скрыли элемент
              }
          }
      

Добавить комментарий

Ваш e-mail не будет опубликован.