Форум программистов, компьютерный форум, киберфорум
C# Windows Forms
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.56/9: Рейтинг темы: голосов - 9, средняя оценка - 4.56
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7

Потоки + Цикл

09.10.2018, 22:02. Показов 1856. Ответов 15
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Добрый вечер столкнулся с проблемой, передо мной стоит задача запускать n потоков в n заходов.
Поясню, например у меня есть 4 потока , которые нужно запустить 4 раза, то есть 4 по 4 (16).

Пытался лепить с двойным циклом, но получается что запускаются сразу ВСЕ (16) , а мне надо что бы выполнились 4, потом 4....
HELP !)
0
Лучшие ответы (1)
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
09.10.2018, 22:02
Ответы с готовыми решениями:

Цикл и потоки
Вот мой примерный код . void Starter() { for ( ;; ) { // тут код } }

Байтовые потоки и потоки символов
Объясните, пожалуйста подробно что имелось ввиду На самом низком уровне все операции ввода/вывода в С# оперируют байтами. Подобный...

Почему если условие входа в цикл не выполняется, функция заново входит в цикл
Сам код: private void traverse_node(TreeControlItem node) { TreeControlItem childNode = default(TreeControlItem);...

15
1 / 1 / 0
Регистрация: 25.11.2017
Сообщений: 37
09.10.2018, 22:12
Может добавить условие. Мол пока не завершится первый поток, второй не начинает работу. Только я не знаю как =)
0
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7
09.10.2018, 22:16  [ТС]
Чисто логически - нужно написать условие что бы цикл дожидался завершения всех потоков, гугл в помощь , но как сформировать код я не знаю, на пальцах понимаю как должно работать, я компьютеру объяснить не могу.
0
1 / 1 / 0
Регистрация: 25.11.2017
Сообщений: 37
09.10.2018, 22:24
Что-то про синхронизацию потоков видел.
0
182 / 145 / 94
Регистрация: 30.07.2013
Сообщений: 579
10.10.2018, 09:43
Лучший ответ Сообщение было отмечено Tarhun_ как решение

Решение

emulzone,

Добавлено через 52 минуты
1. Добавьте на форму кнопку btnStart и подпишитесь на событие Click
2. Добавьте на форму текстбокс tbOutput с свойством Multiline = True и VerticalScrollbar = True
3. Код под спойлером
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Linq;
using System.Threading;
using System.Windows.Forms;
 
namespace emulzone
{
    public partial class Form1 : Form
    {
 
        List<BackgroundWorker> BwPool;
 
        public Form1()
        {
            InitializeComponent();
        }
 
        private void Form1_Load(object sender, EventArgs e)
        {
            BwPool = new List<BackgroundWorker>()
            {   
                new BackgroundWorker(),
                new BackgroundWorker(),
                new BackgroundWorker(),
                new BackgroundWorker()
            };
            foreach (BackgroundWorker bw in BwPool)
            {
                bw.DoWork += BW_DoWork;
                bw.RunWorkerCompleted += BW_WorkComplete;
            }
        }
 
        private void btnStart_Click(object sender, EventArgs e)
        {
            tbOutput.Clear();
            tbOutput.AppendText("TEST STARTED\n");
            for (int i = 1; i <= 4; i++)
            {
                tbOutput.AppendText(String.Format("Iteration {0}\n", i));
                foreach (BackgroundWorker bw in BwPool)
                {
                    bw.RunWorkerAsync();
                }
                while(BwPool.Where(x => x.IsBusy).Count() > 0)
                {
                    Thread.Sleep(10);
                    Application.DoEvents();
                }
            }
            tbOutput.AppendText("TEST COMPLETED");
        }
 
        private void BW_WorkComplete(object sender, RunWorkerCompletedEventArgs e)
        {
            tbOutput.Invoke((MethodInvoker)delegate () { tbOutput.AppendText(String.Format("BW {0} completed\n", BwPool.IndexOf(sender as BackgroundWorker))); });
        }
 
        private void BW_DoWork(object sender, DoWorkEventArgs e)
        {
            tbOutput.Invoke((MethodInvoker)delegate () { tbOutput.AppendText(String.Format("BW {0} started\n", BwPool.IndexOf(sender as BackgroundWorker))); });
            Random rnd = new Random();
            Thread.Sleep(rnd.Next(1000, 10000));
        }
    }
}


Пример выполнения программы:
Кликните здесь для просмотра всего текста
Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
TEST STARTED
Iteration 1
BW 0 started
BW 3 started
BW 1 started
BW 2 started
BW 0 completed
BW 3 completed
BW 1 completed
BW 2 completed
Iteration 2
BW 1 started
BW 2 started
BW 3 started
BW 0 started
BW 2 completed
BW 3 completed
BW 0 completed
BW 1 completed
Iteration 3
BW 0 started
BW 1 started
BW 3 started
BW 2 started
BW 0 completed
BW 1 completed
BW 3 completed
BW 2 completed
Iteration 4
BW 1 started
BW 2 started
BW 3 started
BW 0 started
BW 1 completed
BW 2 completed
BW 3 completed
BW 0 completed
TEST COMPLETED
1
 Аватар для V_Monomax
1406 / 1260 / 20
Регистрация: 09.08.2011
Сообщений: 2,319
Записей в блоге: 1
10.10.2018, 09:55
Tarhun_, копай в сторону пула потоков, пока не закончился пул потоков 1 (4 потока), не начинать другой.
1
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7
10.10.2018, 11:45  [ТС]
Спасибо за ответ ! А как сделать что бы кол-во было изменяемым, просто цикл для списка не работает, я про часть кода:

C#
1
2
3
4
5
6
7
8
9
10
11
12
13
private void Form1_Load(object sender, EventArgs e)
        {
         //.......
            BwPool = new List<BackgroundWorker>()
            {   
                new BackgroundWorker(),
                new BackgroundWorker(),
                new BackgroundWorker(),
                new BackgroundWorker()
            };
            
        //.......
        }
Кликните здесь для просмотра всего текста

C#
1
2
3
4
for (int i = 1; i <= 4; i++)
            {
                BwPool = new List<BackgroundWorker>() { new BackgroundWorker() };
            }



Кликните здесь для просмотра всего текста

Старт:
Iteration 1
BW 0 started
BW 0 completed
Iteration 2
BW 0 started
BW 0 completed
Iteration 3
BW 0 started
BW 0 completed
Iteration 4
BW 0 started
BW 0 completed
TEST COMPLETED
0
 Аватар для V_Monomax
1406 / 1260 / 20
Регистрация: 09.08.2011
Сообщений: 2,319
Записей в блоге: 1
10.10.2018, 13:39
Цитата Сообщение от Tarhun_ Посмотреть сообщение
А как сделать что бы кол-во было изменяемым
не понял вопроса, ведь вы уже создали изменяемый List<BackgroundWorker>(), правда, сделали это очень плохо никогда, слышите, никогда, не используйте логику в обработчиках контролов. Для того что-бы все работало правильно, разделите логику на две части Логика работы с данными в данной части у вас не должно быть хоть сколько-нибудь упоминания контролов, и соответственно у вас должна быть Логика работы контролов, где вы обращаетесь к Логике работы с данными, и в зависимости от результатов меняете состояния контролов.
Тогда будет легче самому писать код, и конечно проще будет вам показать где ошибка. Пока в вашей логике, не видно проблемы, так как логика написана плохо, и еще хуже сформулирован вопрос.
1
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7
10.10.2018, 13:53  [ТС]
Тов. meridbt предоставил рабочий код , который меня почти утаивает. Он сделал что можно изменять количество заходов (пулов), но нет возможности изменить кол-во создаваемых потоков ( из программы ), он вызывает их через код:
C#
1
new BackgroundWorker()
, я не нашел подробной литературы как сделать это циклично (в моем варианте запускаеться только 1 поток).

Кол-во операций\Кол-во потоков
операция 0 Поток 0 Поток 1 Поток 2 Поток m
операция 1 Поток 0 Поток 1 Поток 2 Поток m
операция 2 Поток 0 Поток 1 Поток 2 Поток m
операция 3 Поток 0 Поток 1 Поток 2 Поток m
................
операция n Поток 0 Поток 1 Поток 2 Поток m
0
 Аватар для V_Monomax
1406 / 1260 / 20
Регистрация: 09.08.2011
Сообщений: 2,319
Записей в блоге: 1
10.10.2018, 14:17
то что показано у meridbt, только пример, ну и потом, там не идет про пул потоков речи.
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public void DoWork()
{
    // Queue a task.
    System.Threading.ThreadPool.QueueUserWorkItem(
        new System.Threading.WaitCallback(SomeLongTask));
    // Queue another task.
    System.Threading.ThreadPool.QueueUserWorkItem(
        new System.Threading.WaitCallback(AnotherLongTask));
}
 
private void SomeLongTask(Object state)
{
    //ваш код для потока
}
 
private void AnotherLongTask(Object state)
{
    // ваш код для потока.
}
Вот это и есть использование пула потоков.
1
182 / 145 / 94
Регистрация: 30.07.2013
Сообщений: 579
10.10.2018, 14:24
Tarhun_,

В этой части программы вы последовательно 4 раза инициализируете коллекцию с одним элементом. После четвёртой (да и любой другой) итерации в коллекции будет только один элемент:

C#
1
2
3
4
for (int i = 1; i <= 4; i++)
            {
                BwPool = new List<BackgroundWorker>() { new BackgroundWorker() };
            }
Попробуйте так:

C#
1
2
3
4
5
6
7
BwPool = new List<BackgroundWorker>();
int threadNum = 4
 
for (int i = 0; i < threadNum; i++)
            {
                BwPool.Add(new BackgroundWorker());
            }
1
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7
10.10.2018, 14:38  [ТС]
Допустил ошибку, все работает, спасибо Всем !
0
182 / 145 / 94
Регистрация: 30.07.2013
Сообщений: 579
10.10.2018, 14:44
=delete=

Добавлено через 4 минуты
Tarhun_,

Хорошо, но, как сказал V_Monomax, это только пример (даже скорее идея реализации), но не готовое решение. Развивайте её в своём проекте и реализуйте грамотно.
0
0 / 0 / 0
Регистрация: 09.10.2016
Сообщений: 7
10.10.2018, 14:51  [ТС]
meridbt , это единственно здравый костыль который подошел для реализации, со временем подробно с этим разберусь и сделаю грамотней
0
182 / 145 / 94
Регистрация: 30.07.2013
Сообщений: 579
10.10.2018, 16:09
V_Monomax,

Цитата Сообщение от V_Monomax Посмотреть сообщение
Вот это и есть использование пула потоков.
подскажите, пожалуйста, как правильно отследить завершение задач, поставленных в очередь таким образом

Класс Worker:
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
class Worker
    {
 
        int threadNum;
        int iterationNum;
        Random rnd;
 
        public delegate void LC(string Value);
        public event LC LogChanged;
 
        private void AppendLog(string value)
        {            
            LogChanged?.Invoke(value);
        }
 
        public Worker(int threadNum, int iterationNum)
        {
            this.threadNum = threadNum;
            this.iterationNum = iterationNum;
            rnd = new Random();
        }
 
        public void DoTest()
        {
            AppendLog(String.Format("TEST STARTED"));
            Task task;
            for (int i = 0; i < iterationNum; i++)
            {
                AppendLog(String.Format("Iteration {0}", i));
                task = DoIteration();
                task.Wait();
            }            
            AppendLog(String.Format("TEST STOPPED"));
        }
 
        private Task DoIteration()
        {
            var task = Task.Run(() =>
            {
                for (int i = 0; i < threadNum; i++)
                {
                    ThreadPool.QueueUserWorkItem(new WaitCallback(DoWorkTask), i);
                    Thread.Sleep(rnd.Next(100, 500));
                }
            });
            task.Wait();
            return task;  
        }
 
        private void DoWorkTask(Object state)
        {
            AppendLog(String.Format("{1} Thread {0} started", state.ToString(), DateTime.Now.ToString("HH:mm:ss")));
            Thread.Sleep(rnd.Next(1000, 5000));
            AppendLog(String.Format("{1} Thread {0} completed", state.ToString(), DateTime.Now.ToString("HH:mm:ss")));
        }
 
    }


На форме:
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
private void btnAlter_Click(object sender, EventArgs e)
        {
            Worker worker = new Worker(4, 1);
            worker.LogChanged += LogChanged;
            worker.DoTest();
        }
 
        private void LogChanged(string value)
        {
            tbOutput.Invoke((MethodInvoker)delegate () { tbOutput.AppendText("\r\n" + value); });
        }


Вывод одной итерации такой:
Кликните здесь для просмотра всего текста
Code
1
2
3
4
5
6
7
8
9
10
11
TEST STARTED
Iteration 0
16:07:17 Thread 0 started
16:07:18 Thread 1 started
16:07:18 Thread 2 started
16:07:19 Thread 3 started
TEST STOPPED
16:07:20 Thread 1 completed
16:07:21 Thread 3 completed
16:07:21 Thread 2 completed
16:07:24 Thread 0 completed
0
 Аватар для V_Monomax
1406 / 1260 / 20
Регистрация: 09.08.2011
Сообщений: 2,319
Записей в блоге: 1
10.10.2018, 18:33
Цитата Сообщение от meridbt Посмотреть сообщение
подскажите, пожалуйста, как правильно отследить завершение задач, поставленных в очередь таким образом
лучше заведите отдельную ветку, а здесь, прочитав наискосок, не имея студии под рукой, могу сказать что я пока не вижу, как вам дождаться завершения потока, тут либо вводить семафор, ну или хотя бы публичный объкт блокировки, и его в основном потоке проверять на блокирование.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
10.10.2018, 18:33
Помогаю со студенческими работами здесь

Помогите заменить цикл forech на простой цикл for
Помогите пожалуйста, срочно. foreach (char xx in strX) { tmp = xx; m--; }...

Цикл foreach заменить на двойной цикл for
int mas = new int; Random Gen = new Random(); for (int i = 0; i &lt; mas.GetLength(0); i++) for...

Foreach цикл и потоки
Список можно читать так: for(auto&amp;pos:list) //bla-bla-bla Поток можно читать так: some_type value; while(stream&gt;&gt;value)...

Создать программу по всем 3 видам циклов...цикл с параметром,цикл с условием,цикл,и цикл с предусловием...
Найти сумму чисел 1 в квадрате до 10 c квадрате...операцию возведению в степень не использовать учесть особенности получения квадратного...

Что такое потоки ввода, потоки вывода?
Здарова всем! Не так давно уже прогаю на С++ и все НИКАК не могу понять, что такое потоки ввода, потоки вывода..! вот допустим...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
16
Ответ Создать тему
Новые блоги и статьи
Оттенки серого
Argus19 18.03.2026
Оттенки серого Нашёл в интернете 3 прекрасных модуля: Модуль класса открытия диалога открытия/ сохранения файла на Win32 API; Модуль класса быстрого перекодирования цветного изображения в оттенки. . .
SDL3 для Desktop (MinGW): Рисуем цветные прямоугольники с помощью рисовальщика SDL3 на Си и C++
8Observer8 17.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-rectangles-sdl3-c. zip finish-rectangles-sdl3-cpp. zip
Символические и жёсткие ссылки в Linux.
algri14 15.03.2026
Существует два типа ссылок — символические и жёсткие. Ссылка в Linux — это запись в каталоге, которая может указывать либо на inode «файла-ИСТОЧНИКА», тогда это будет «жёсткая ссылка» (hard link),. . .
[Owen Logic] Поддержание уровня воды в резервуаре количеством включённых насосов: моделирование и выбор регулятора
ФедосеевПавел 14.03.2026
Поддержание уровня воды в резервуаре количеством включённых насосов: моделирование и выбор регулятора ВВЕДЕНИЕ Выполняя задание на управление насосной группой заполнения резервуара,. . .
делаю науч статью по влиянию грибов на сукцессию
anaschu 13.03.2026
прикрепляю статью
SDL3 для Desktop (MinGW): Создаём пустое окно с нуля для 2D-графики на SDL3, Си и C++
8Observer8 10.03.2026
Содержание блога Финальные проекты на Си и на C++: hello-sdl3-c. zip hello-sdl3-cpp. zip Результат:
Установка CMake и MinGW 13.1 для сборки С и C++ приложений из консоли и из Qt Creator в EXE
8Observer8 10.03.2026
Содержание блога MinGW - это коллекция инструментов для сборки приложений в EXE. CMake - это система сборки приложений. Здесь описаны базовые шаги для старта программирования с помощью CMake и. . .
Как дизайн сайта влияет на конверсию: 7 решений, которые реально повышают заявки
Neotwalker 08.03.2026
Многие до сих пор воспринимают дизайн сайта как “красивую оболочку”. На практике всё иначе: дизайн напрямую влияет на то, оставит человек заявку или уйдёт через несколько секунд. Даже если у вас. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru