Форум по Delphi программированию

Delphi Sources



Вернуться   Форум по Delphi программированию > Все о Delphi > Программа и интерфейс
Ник
Пароль
Регистрация <<         Правила форума         >> FAQ Пользователи Календарь Поиск Сообщения за сегодня Все разделы прочитаны

Ответ
 
Опции темы Поиск в этой теме Опции просмотра
  #1  
Старый 12.03.2014, 04:54
novashdima novashdima вне форума
Новичок
 
Регистрация: 05.02.2010
Адрес: Украина, Киев
Сообщения: 64
Версия Delphi: XE3, XE4
Репутация: 10
По умолчанию Работа с потоками

Наверняка уже много раз тема обсуждалась, но хочу попросить совета по правильной реализации задачи в словах и примерах кода.
Итак, программа занимается обработкой данных, это может занять продолжительное время, поэтому нужна кнопка приостановки процесса.
Во время обработки данных происходит изменение компонентов Gauge и Memo.
Открыв проект в новой версии студии получаю ошибку при компилировании, последняя ли это версия проекта и было ли что-то после нормальной сборки непонятно (давно занимался проектом).
Итак, вопросы по реализации:
при остановке процесса поток приостанавливать с возможностью продолжения или полной остановки, в первом случае вызываем resume, во втором Terminate?
в данный момент компилятор ругается на функцию синхронизации.
Напишите пример взаимодействия, описания и выполнения действий, у меня пока так реализовано:
Код:
type
  MyThread = class(TThread)
  protected
    CriticalSection: TCriticalSection;
    procedure UpdateMemo(Text: string);
    procedure UpdateGauge(Position: Integer);
    procedure Execute; override;
    procedure Analys;
  end;

var
AnalysThread: MyThread;

procedure TForm1.FormCreate(Sender: TObject);
begin
  try
    AnalysThread := MyThread.Create(True);
    AnalysThread.FreeOnTerminate := False;
    AnalysThread.Priority := tpLower;
    //.....
end;

procedure TForm1.StartProcessClick(Sender: TObject);
begin
  AnalysThread.Resume;
  //.....
end;

procedure TForm1.StopProcessClick(Sender: TObject);
begin
  AnalysThread.Terminate;
  //....
end;

procedure MyThread.Analys;
begin
  try
   if AnalysThread.Terminated then Exit;
   CriticalSection.Enter;
   //....
   Synchronize(UpdateMemo(StringList.Text));
   //....
  finally
  CriticalSection.Leave;
  end;
end;

procedure MyThread.Execute;
begin
  inherited;
  if not AnalysThread.Terminated then
    AnalysThread.Synchronize(Analys);
end;

procedure MyThread.UpdateGauge(Position: Integer);
begin
  Form1.Gauge.Progress := Position;
end;

procedure MyThread.UpdateMemo(Text: string);
begin
  Form1.Memo.Text := Text;
  Form1.Memo.Refresh;
end;

Для начала имхо надо вместо двух кнопок для начала/остановки обработки сделать одну и проверять состояние потока (Suspend/Terminate?).
И мне не нравится взаимодействие с потоками в процедуре анализа... в общем внимательно выслушаю предложения по приведению кода в нормальный вид и главное рабочий.

Последний раз редактировалось novashdima, 12.03.2014 в 06:05.
Ответить с цитированием
  #2  
Старый 12.03.2014, 10:01
icWasya icWasya вне форума
Местный
 
Регистрация: 09.11.2010
Сообщения: 499
Репутация: 10
По умолчанию

во первых
Код:
procedure MyThread.Execute; 
begin  
  while not Terminated do
     Analys; 
  end; 
end;

во вторых, у Synchronize параметром должен быть метод без параметров.
В третьих - где
Код:
 CriticalSection:= TCriticalSection.Create;
?
В четвёртых - что с чем Вы пытаетесь синхронизировать с помощью этой критической секции? Если разные потоки, то для них эта секция должна быть одна, а не по секции в каждом потоке.
Ответить с цитированием
  #3  
Старый 12.03.2014, 13:39
novashdima novashdima вне форума
Новичок
 
Регистрация: 05.02.2010
Адрес: Украина, Киев
Сообщения: 64
Версия Delphi: XE3, XE4
Репутация: 10
По умолчанию

Цитата:
Сообщение от icWasya
во первых
Код:
procedure MyThread.Execute; 
begin  
  while not Terminated do
     Analys; 
  end; 
end;
ок, так и сделаю
Цитата:
Сообщение от icWasya
во вторых, у Synchronize параметром должен быть метод без параметров.
Ок, сделал стринглист глобальным. Пытаюсь сделать вызов Synchronize(UpdateMemo); получаю E2250 There is no overloaded version of 'Synchronize' that can be called with these arguments
Цитата:
Сообщение от icWasya
В третьих - где
Код:
 CriticalSection:= TCriticalSection.Create;
?
Забыл вставить сюда, есть в создании формы.
Цитата:
Сообщение от icWasya
В четвёртых - что с чем Вы пытаетесь синхронизировать с помощью этой критической секции? Если разные потоки, то для них эта секция должна быть одна, а не по секции в каждом потоке.
Не знаю, почему сделал именно так. Перенес в основной поток.

Последний раз редактировалось novashdima, 12.03.2014 в 14:00.
Ответить с цитированием
  #4  
Старый 12.03.2014, 14:55
Аватар для poli-smen
poli-smen poli-smen вне форума
Профессионал
 
Регистрация: 06.08.2012
Адрес: Кривой Рог
Сообщения: 1,791
Версия Delphi: Delphi 7, XE2
Репутация: 4415
По умолчанию

Цитата:
Сообщение от novashdima
Ок, сделал стринглист глобальным.
Параметры для "синхронизированного" метода обычно передают через поля объекта потока (который здесь MyThread).
Цитата:
Сообщение от novashdima
Пытаюсь сделать вызов Synchronize(UpdateMemo); получаю E2250 There is no overloaded version of 'Synchronize' that can be called with these arguments
А параметр "Text: string" из метода UpdateMemo точно убрал?
Ответить с цитированием
  #5  
Старый 12.03.2014, 15:04
novashdima novashdima вне форума
Новичок
 
Регистрация: 05.02.2010
Адрес: Украина, Киев
Сообщения: 64
Версия Delphi: XE3, XE4
Репутация: 10
По умолчанию

Цитата:
Сообщение от poli-smen
Параметры для "синхронизированного" метода обычно передают через поля объекта потока (который здесь MyThread).
Подробней пожалуйста. Имеется в виду Synchronize(MyThread.UpdateMemo) ?
Цитата:
Сообщение от poli-smen
А параметр "Text: string" из метода UpdateMemo точно убрал?
Точно
Ответить с цитированием
  #6  
Старый 12.03.2014, 15:23
Аватар для Alegun
Alegun Alegun вне форума
LMD-DML
 
Регистрация: 12.07.2009
Адрес: Богородское
Сообщения: 3,025
Версия Delphi: D7E
Репутация: 1834
По умолчанию

Вероятно имелся ввиду классический вариант "своего" потока, типа как в примере из DRKB
Код:
type
TMyThread = class(TThread)
 private
  Answer: integer;
  protected
   procedure ShowResult;
   procedure Execute; override;
end;

implementation

//Процедура для вывода информации из потока
procedure TMyThread.ShowResult;
begin
 Form1.Memo.Text:= IntToStr(Answer);
 Form1.Memo.Refresh;
end;

procedure TMyThread.Execute;
var
i: Integer;
begin
for i := 1 to 10000 do
begin
   Inc(answer);

   Synchronize(ShowResult);

end;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
MyThread: TMyThread;
begin
MyThread:= TMyThread.Create(false);
end;
Ответить с цитированием
Ответ


Delphi Sources

Опции темы Поиск в этой теме
Поиск в этой теме:

Расширенный поиск
Опции просмотра
Комбинированный вид Комбинированный вид

Ваши права в разделе
Вы не можете создавать темы
Вы не можете отвечать на сообщения
Вы не можете прикреплять файлы
Вы не можете редактировать сообщения

BB-коды Вкл.
Смайлы Вкл.
[IMG] код Вкл.
HTML код Выкл.
Быстрый переход


Часовой пояс GMT +3, время: 02:19.


 

Сайт

Форум

FAQ

RSS лента

Прочее

 

Copyright © Форум "Delphi Sources" by BrokenByte Software, 2004-2023

ВКонтакте   Facebook   Twitter