Форум программистов, компьютерный форум, киберфорум
C++ Builder
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.50/4: Рейтинг темы: голосов - 4, средняя оценка - 4.50
Заблокирован

Странная работа потока

18.06.2015, 23:55. Показов 794. Ответов 4
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Вообщем не могу понять в чем дело, есть программа:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//---------------------------------------------------------------------------
void __fastcall TForm1::Execute() {
        TGenThread *GenThread = new TGenThread(true);
        GenThread->Priority = tpHigher;
        GenThread->Resume();
        Sleep(400);
        for ( int i = 1 ; i < 5 ; ++i ) {
                //GenThread->Resume();
                Form1->Memo1->Lines->Add(IntToStr(i));
        }
        GenThread->Terminate();
        GenThread->Resume();
}
//---------------------------------------------------------------------------
 
void __fastcall TForm1::Button1Click(TObject *Sender)
{
Execute();        
}
и поток :
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
void __fastcall TGenThread::Execute()
{
        while(1) {
                if(Terminated) break;
                Synchronize(Update);
                Suspend();
        }
}
//---------------------------------------------------------------------------
void  __fastcall TGenThread::Update()
{
        Form1->Memo1->Lines->Add( "yyy" );
}
ожидал получить в результате чтото вроде:
yyy
1
2
3
4
а получил:
1
2
3
4
yyy
почему поток срабатывает после цикла? хотя специально поставил его вначале и даже сделал sleep?
изначально необходимо было чтобы поток запускался в цикле, а после выполнения приостанавливался до следующей итерации цикла, но результат был аналогичен (1,2,3,4,yyy)
версия билдера 6
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
18.06.2015, 23:55
Ответы с готовыми решениями:

Странная работа TTimer
Здраствуйте) Есть два таймера: у первого интервал 1 (т.е. 1/1000) секунды, у второго интервал 1000. когда срабатывает...

Странная работа QNetworkProxy
Нашел странное поведение компоненты QNetworkProxy. В одной и той же организации, в одной и той же доменной сети работают два компа с...

Странная работа QMediaPlayer
Здравствуйте! Я сейчас пишу простенький плеер: window.h #ifndef WINDOW_H #define WINDOW_H

4
 Аватар для Fulcrum_013
2083 / 1574 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
19.06.2015, 00:13
Synchronize causes the call specified by AMethod to be executed using the main thread, thereby avoiding multithread conflicts
- т.е. TGenThread::Update() отрабатывает в главном потоке а не в потоке GenThread;
0
1408 / 572 / 127
Регистрация: 31.10.2011
Сообщений: 1,960
19.06.2015, 05:07
интересный способ запуска потока
и Sleep(400) работает не в потоке

сделай функции работы с потоком (обертки), не принадлежащие классу потока, но в модуле потока
и безопасно вызывай их с любой формы, но Synchronize в потоке нужно делать полюбому

Добавлено через 17 минут
C++
1
2
3
4
5
6
7
8
9
10
11
12
void __fastcall TForm1::Execute() {
        TGenThread *GenThread = new TGenThread(true);
        GenThread->Priority = tpHigher;
        GenThread->Resume();  // ???? у  тебя и так TGenThread(true), то есть создавай TGenThread(false);
        Sleep(400);  // зачем ?
        for ( int i = 1 ; i < 5 ; ++i ) {
                //GenThread->Resume();
                Form1->Memo1->Lines->Add(IntToStr(i)); // это нужно делать через Synchronize 
        }
        GenThread->Terminate();
        GenThread->Resume(); // это что? после убийства потока нет смысла его дергать
}
0
20 / 20 / 7
Регистрация: 18.02.2015
Сообщений: 304
19.06.2015, 18:14
Слушай, а можешь кинуть пример создания данного потока. Типо

поток {
//Тут описывается инструкция функции которая будет работать в потоке
}

Где-то в программе {
Запустили поток
}
0
Заблокирован
20.06.2015, 02:01  [ТС]
Переписал чутка прогу, теперь при нажатии кнопки создаются 3 потока, один из которых является поставщиком данных для двух других, которые обрабатывают эти данные и результаты заносят в список listData.

Вообщем вот что получилось:
Основной поток:
Кликните здесь для просмотра всего текста
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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
//---------------------------------------------------------------------------
#include <list.h>
#include <vcl.h>
#pragma hdrstop
 
#include "Thread.h"
#include "ClassThread.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm1 *Form1;
CRITICAL_SECTION cs;
TGenThread *MainThread;
TGenThread *GenThread;
TGenThread *GenThread1;
//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
        : TForm(Owner)
{
        InitializeCriticalSection(&cs);
 
}
//---------------------------------------------------------------------------
void __fastcall TForm1::CreateThread() {
        terminate = 0; // сбрасываем флаг завешения работы
        //создаем три потока
        MainThread = new TGenThread(true);
        GenThread = new TGenThread(true);
        GenThread1 = new TGenThread(true);
* * * * //инициализируем их
        MainThread->Priority = tpHigher;
        MainThread->Init(0);
        GenThread->Priority = tpHigher;
        GenThread->Init(1);
        GenThread1->Priority = tpHigher;
        GenThread1->Init(2);
* * * * //запускаем главный поток (поставщик данных для двух других)
        MainThread->Resume();
}
//---------------------------------------------------------------------------
void __fastcall TForm1::Execute() {
 
        for ( int i = 1 ; i < 5 ; ++i ) {
                        flag = 0; // сброс флага завершенных потоками вычислений
                        it = i; // заносим в глобальную переменную новые данные для обработки
                        GenThread->Resume(); // запускаем поток 1
                        GenThread1->Resume(); // запускаем поток 2
                        MainThread->Suspend(); // останавливаем главный поток
        }
        terminate = 1; // выставляем флаг завершения работы
        GenThread->Resume(); // запускаем выход цикла в поток 1 
        GenThread1->Resume(); // запускаем выход цикла в поток 2  
}
//---------------------------------------------------------------------------
void __fastcall TForm1::Generation() {
                        while (!terminate) { 
                        EnterCriticalSection(&cs);
                                listData.push_back(it+10); // обработка данных
                                flag++; // увеличиваем флаг отработанных потоков
                        LeaveCriticalSection(&cs);
                        if ( flag == 2 ) MainThread->Resume(); // проверяем флаг, если этот поток завершил вычисления последним, то запускаем главный поток
                        GenThread->Suspend(); // останавливаем поток 1
                        }
}
//---------------------------------------------------------------------------
void __fastcall TForm1::Generation1() {
                        while (!terminate) {
                        EnterCriticalSection(&cs);
                                listData.push_back(it+100); // обработка данных
                                flag++;  // увеличиваем флаг отработанных потоков
                        LeaveCriticalSection(&cs);
                        if ( flag == 2 ) MainThread->Resume();  // проверяем флаг, если этот поток завершил вычисления последним, то запускаем главный поток
                        GenThread1->Suspend(); // останавливаем поток 1
                        }
}
//---------------------------------------------------------------------------
 
void __fastcall TForm1::Button1Click(TObject *Sender)
{
        CreateThread(); // создание потоков по нажатию на кнопке
}
//---------------------------------------------------------------------------
 
void __fastcall TForm1::Button2Click(TObject *Sender)
{
        Memo1->Lines->Add("-----");
        for ( list<int>::iterator it = listData.begin() ; it != listData.end() ; ++it ) {
                Memo1->Lines->Add(*it);
        }
}
//---------------------------------------------------------------------------
 
void __fastcall TForm1::FormClose(TObject *Sender, TCloseAction &Action)
{
        DeleteCriticalSection(&cs);
}
//---------------------------------------------------------------------------

Потоки обработки:
Кликните здесь для просмотра всего текста
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//---------------------------------------------------------------------------
void __fastcall TGenThread::Execute()
{
        FreeOnTerminate = true; //флаг освобождения памяти при завершении потока
        // запуск функции в потоке в зависимости от ключа инициализации
        if (NumF == 0) {
                Form1->Execute(); //
        } else if (NumF == 1) {
                Form1->Generation();
        } else if (NumF == 2) {
                Form1->Generation1();
        } else {};
}
 
//---------------------------------------------------------------------------
void  __fastcall TGenThread::Init( int NumF_)
{
        NumF = NumF_; // Создаем ключ для запуска
}


Добавлено через 16 минут
Цитата Сообщение от _Dimon_ Посмотреть сообщение
интересный способ запуска потока
и Sleep(400) работает не в потоке
слип нужен был для торможения основного потока, чтобы поток который я создал сработал раньше чем основной, но это не помогло.
Про Ваши комментарии в коде:
Цитата Сообщение от _Dimon_ Посмотреть сообщение
GenThread->Resume(); *// ???? у *тебя и так TGenThread(true), то есть создавай TGenThread(false);
TGenThread(true) - создает поток в приоставновленом состоянии, а не наоборот
Цитата Сообщение от _Dimon_ Посмотреть сообщение
Form1->Memo1->Lines->Add(IntToStr(i)); // это нужно делать через Synchronize
Это строка находится в TForm1::Execute() тоесть выполняется в основном потоке программы, никаких синхронизаций там нет.
Цитата Сообщение от _Dimon_ Посмотреть сообщение
GenThread->Terminate();
* * * * GenThread->Resume(); // это что? после убийства потока нет смысла его дергать
GenThread->Terminate() не останавливает поток, а выставляет флаг, что поток можно завершить, учитывая что в этот момент поток находится на паузе, его нужно возобновить чтоб эта строка отработала.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
20.06.2015, 02:01
Помогаю со студенческими работами здесь

Странная работа getch
#include &lt;iostream&gt; #include &lt;conio.h&gt; using namespace std; int main() { int m; m=getch(); switch (m)

Странная работа FindFirstFile
Здравствуйте! Вывожу функциями FindFirstFile и FindNextFile и получаю имена &quot;.&quot; и &quot;..&quot; далее идут все файлы находящиеся в указанной...

Странная работа Wi-Fi
Вчера начал пропадать интернет т.е. есть, рраз на пол минуты - минуту отключается и снова есть, пропадет по разному бывает каждые 5 минут,...

Странная работа ПК
Всем привет. Вчера все было нормально, сегодня начал заходить в игру - начал замечать что при заходе в нее начинает что-то свистеть в...

Странная работа БД
Спасибо всем тем, кто решил откликнуться. Сначала лирика: Имеется проект &quot;Калькулятор&quot;, созданы различные типы инженерный,...


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

Или воспользуйтесь поиском по форуму:
5
Ответ Создать тему
Новые блоги и статьи
http://iceja.net/ математические сервисы
iceja 20.01.2026
Обновила свой сайт http:/ / iceja. net/ , приделала Fast Fourier Transform экстраполяцию сигналов. Однако предсказывает далеко не каждый сигнал (см ограничения http:/ / iceja. net/ fourier/ docs ). Также. . .
http://iceja.net/ сервер решения полиномов
iceja 18.01.2026
Выкатила http:/ / iceja. net/ сервер решения полиномов (находит действительные корни полиномов методом Штурма). На сайте документация по API, но скажу прямо VPS слабенький и 200 000 полиномов. . .
Расчёт переходных процессов в цепи постоянного тока
igorrr37 16.01.2026
/ * Дана цепь постоянного тока с R, L, C, k(ключ), U, E, J. Программа составляет систему уравнений по 1 и 2 законам Кирхгофа, решает её и находит переходные токи и напряжения на элементах схемы. . . .
Восстановить юзерскрипты Greasemonkey из бэкапа браузера
damix 15.01.2026
Если восстановить из бэкапа профиль Firefox после переустановки винды, то список юзерскриптов в Greasemonkey будет пустым. Но восстановить их можно так. Для этого понадобится консольная утилита. . .
Сукцессия микоризы: основная теория в виде двух уравнений.
anaschu 11.01.2026
https:/ / rutube. ru/ video/ 7a537f578d808e67a3c6fd818a44a5c4/
WordPad для Windows 11
Jel 10.01.2026
WordPad для Windows 11 — это приложение, которое восстанавливает классический текстовый редактор WordPad в операционной системе Windows 11. После того как Microsoft исключила WordPad из. . .
Classic Notepad for Windows 11
Jel 10.01.2026
Old Classic Notepad for Windows 11 Приложение для Windows 11, позволяющее пользователям вернуть классическую версию текстового редактора «Блокнот» из Windows 10. Программа предоставляет более. . .
Почему дизайн решает?
Neotwalker 09.01.2026
В современном мире, где конкуренция за внимание потребителя достигла пика, дизайн становится мощным инструментом для успеха бренда. Это не просто красивый внешний вид продукта или сайта — это. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru