|
|
Регистрация | << Правила форума >> | FAQ | Пользователи | Календарь | Поиск | Сообщения за сегодня | Все разделы прочитаны |
|
Опции темы | Поиск в этой теме | Опции просмотра |
#16
|
||||
|
||||
Еще пара вопросов возникла:
Назначаю процедуру для OnTerminate создаваемого потока Код:
procedure TForm1.PoolCeate; var i: Byte; begin SetLength(FPool,edtPoolCount.Value); for i := Low(FPool) to High(FPool) do begin FPool[i]:= TDLThrd.Create(DlMode,edtURL.Text,Queue,WriteResult); {FreeOnTerminate:= True} TDLThrd(FPool[i]).OnTerminate:= Form1.ThrTerminate; TDLThrd(FPool[i]).Start; end; end; procedure TForm1.ThrTerminate(Sender: TObject); begin CS.Enter; ListBox1.Items.Append('Поток - '+IntToStr(TDLThrd(Sender).ThreadID)+' - сдох'); CS.Leave; end; Код:
.... if Assigned(TDLThrd(FPool[i])) then ... Код:
.... if WaitForSingleObject(MyThread.Handle, 1000) <> WAIT_OBJECT_0 then ..... Код:
.... if WaitForSingleObject(TDLThrd(FPool[i]).Handle, 1000) = WAIT_FAILED then ..... Но все это мои догадки, хотелось бы услышать мнение специалиста. И если все так, корректен ли будет такой вариант Код:
...... TDLThrd(FPool[i]).OnTerminate:= Form1.ThrTerminate; ..... procedure TForm1.ThrTerminate(Sender: TObject); begin CS.Enter; ListBox1.Items.Append('Поток - '+IntToStr(TDLThrd(Sender).ThreadID)+' - сдох'); CS.Leave; TDLThrd(Sender):= nil; //вариант: Sender:= nil; end; Второй вопрос: поток сначала ждет появления задания, далее выполняет их пока список не опустеет и потом самоуничтожается Код:
procedure TDLThrd.Execute; begin while not FTaskExist do //Ожидание задачи begin ...... end; while FTaskExist do //Выполнение begin ..... end; end; Можно конечно попробовать заранее просчитать количество заданий и не создавать лишних потоков, но все же, хотя бы для общего развития, как убить такой поток? Если не ошибаюсь Free вызывает Terminate и WaitFor, а если так то поток все равно останется жить. Вроде бы есть WinAPI функция, но есть мнение что этот способ не кошерный. |
#17
|
|||
|
|||
тогда уж пиши полноценный менеджер потоков, который сам будет эа всем следить (фактически, еще один поток).
ЗЫ. Да, вызов Free не обнуляет указатель, а уничтожает только сам поток. Более того, если ты почитаешь как устроены потоки в WinAPI, то поймешь, что то, что ты имеешь в Delphi не есть на самом деле сам поток, а всего лишь класс-обертка, предоставляющая удобный интерфейс к потоку. Т.е. сам поток может "умереть", а класс все еще существовать, только большинство вызовов буду оканчиваться исключением. |
Этот пользователь сказал Спасибо lmikle за это полезное сообщение: | ||
Помидоркин (05.01.2019)
|
#18
|
||||
|
||||
Цитата:
Но может быть и ситуация когда класс-обертка все еще жив и обращение TDLThrd(FPool[i]) указывает на вполне себе реальный объект, а самого потока уже нет? Если так, то как узнать что поток завершился, и убить управлявший им объект? |
#19
|
|||
|
|||
Ну, например, в ситуации, когда ты сохранил указатель на поток, но указал ему FreeOnTerminate. Когда поток завершится, то указатель все еще будет указывать на какой-то адрес, а вот потока, как и обертки, уже не будет. Ну и в какой-то момент обращение по этому указателю выдаст AV (не сразу, пока память не будет алоцирована в процессе работы программы, вызовы даже буду срабатывать, по крайней мере некоторые).
Как этого избежать? Технически - никак. Это плата за эфективность. Ровно та же беда в C/C++. В managed языках с этим проще, но и в результате эфективность ниже. Т.О. код надо писать с оглядкой на эти особенности. На самом деле это не так сложно, как кажется. |
Эти 2 пользователя(ей) сказали Спасибо lmikle за это полезное сообщение: | ||
slaidouh77 (20.02.2019),
Помидоркин (08.01.2019)
|