Дополнительные возможности потоков

 

Теперь мы дадим краткий обзор ещё некоторых свойств и методов объекта потока и сделаем несколько замечаний по работе с потоком. В принципе, у объекта не так уж и много свойств и методов, и большую часть возможностей мы уже рассмотрели, но всё же мы обязаны сделать ещё несколько замечаний.

Suspend - приостанавливает поток. Для вызова просто напишите co.Suspend. Чтобы возобновить работу с этой же точки нужно вызвать Resume.

Priority- устанавливает приоритет потока. Например Priority:=tpIdle;

tpIdle - поток будет работать, только когда процессор не занят другими процессами.

tpLowest - самый слабый приоритет

tpLower - слабый приоритет

tpNormal - нормальный

tpHigher - высокий

tpHighest - самый высокий

tpTimeCritical - критичный (мы не советуем использовать, потому что может зависнуть система).

Suspended - если этот параметр true, то поток находится в паузе.

Terminated - если true, то поток должен быть остановлен, иначе поток должен продолжать работу.

Terminate – остановить выполнение потока.

FreeOnTerminate – если это свойство равно true, то по завершении выполнения процедуры Execute поток самоуничтожится. Советуем использовать этот параметр, чтобы быть уверенным в том, что поток корректно удалился из памяти.

Давайте посмотрим, как будет выглядеть наш метод Execute из предыдущего примера с использованием свойства FreeOnTerminate:

 

procedure TCountObj.Execute;

begin

FreeOnTerminate:=true;

index:=1;

//Запускаем бесконечный счётчик

while index>0 do

begin

Synchronize(UpdateLabel);

Inc(index);

if index>100000 then

index:=0;

//Если поток остановлен, то выйти.

if terminated then exit;

end;

end;

 

В принципе, этой информации вам будет достаточно для написания собственных потоков. У объекта TThread есть ещё несколько свойств и методов, но они не так важны.

Объекты потоков создаются как полноценные объекты. В основной программе мы создаём в памяти отдельный экземпляр потока и потом работаем с ним. Вы можете создавать по несколько экземпляров одного потока и они будут работать одновременно, абсолютно не мешая друг другу. Представим пример программы, копирующей файлы. Вы можете создать поток, который будет копировать файлы из одного места в другое. В основной программе можно создать два экземпляра таких потоков и каждому из них задать копирование разных файлов в разные места. Оба потока будут копировать свои файлы, абсолютно не мешая друг другу.

 

Подробней о синхронизации

 

В предыдущем примере мы использовали процедуру UpdateLabel, в которой на главную форму выводится значение переменной index. Если бы мы программировали главное окно, то вполне логичным было бы создать переменную index локальной для процедуры Execute, а её значение передавать в UpdateLabel в качестве параметра. В потоках с этим проблема. Чтобы передать какие-то значения в процедуру, которая должна вызываться методом Synchronize, нужно пользоваться переменными объекта. Мы даже не советуем пробовать передавать параметры в процедуры, которые вызываются методом Synchronize.

Но использование синхронизации – не единственный способ обновления параметров окна. Мы можем использовать для этого событийную модель Windows. Каждый раз, когда надо обновить содержимое текста, мы можем посылать окну сообщение SendMessage с указанием значения, которое надо установить. Главное окно будет получать это сообщение и компонент сам изменит заголовок. В этом случае мы не обращаемся к главному окну из потока, а только отправляем сообщение, поэтому никаких проблем не будет.

Итак, функция SendMessage имеет следующие параметры:

1. Указатель на окно (компонент), которому нужно послать сообщение.

2. Тип сообщения.

3. Первый параметр.

4. Второй параметр.

Судя по функции, нам нужен компонент, у которого есть свойство Handle. В предыдущем примере у нас был TLabel, у которого нет такого свойства, значит, он нам не подходит. Заменим этот компонент на TEdit. Теперь перейдём в поток. Тут в разделе usesнужно добавить два модуля: windows (здесь объявлена сама функция) и messages (здесь находятся все типы сообщений Windows).

Теперь удаляйте из потока процедуру UpdateLabel, больше она не нужна, потому что мы не будем использовать метод Synchronize. Ну и наконец, подкорректируем наш метод Execute:

 

procedure TCountObj.Execute;

begin

index:=1;

while index>0 do

begin

SendMessage(Form1.Edit1.Handle, WM_SETTEXT, 0,

Integer(PChar(IntToStr(index))));

Inc(index);

if index>100000 then

index:=0;

if terminated then exit;

end;

end;

 

Как видите, теперь у нас вместо метода Synchronize генерируется событие на обновления компонента TEdit. В качестве второго параметра мы указываем тип сообщения WM_SETTEXT – обновить информацию. Третий параметр равен нулю. В последнем параметре нужно указать значение, которое нужно установить. Вот тут есть небольшая сложность. У нас значение представлено в виде целого числа, но нужно превратить его в PChar. Для этого мы сначала конвертируем переменную index в строку (IntToStr), потом приводим его к типу PChar и тут же указываем размер Integer.

 

 

ЛАБОРАТОРНОЕ ЗАДАНИЕ

И МЕТОДИЧЕСКИЕ УКАЗАНИЯ

ПО ЕГО ВЫПОЛНЕНИЮ

 

Лабораторное задание

Согласно варианту задания, полученному при выполнении лабораторной работы № 1, написать программу, которая на основе базы данных создает отчет в Word, используя поток при создании документа в Word.