Вызов метода по имени или по адресу
В этом разделе мы рассмотрим различные способы вызова методов фоновой задачи.
Давайте рассмотрим это на примере уже знакомого нам метода SetTimer интерфейса IOmniTaskControl, который мы вкратце рассмотрели в разделе “Двухсторонняя связь. Подход FWorker := TAsyncHello.Create('Hello');
FHelloTask :=
OmniEventMonitor1.Monitor(CreateTask(FWorker, 'Hello'))
.SetTimer(1000, MSG_SEND_MESSAGE) // ПОСЫЛАЕМ СООБЩЕНИЕ
.SetParameter('Delay', 1000)
.Run;
После чего в фоновой задаче мы перехватывали это сообщение обработчиком
TAsyncHello = class(TOmniWorker)
strict private
aiCount : integer;
aiMessage: string;
public
. . .
// Наш обработчик MSG_SEND_MESSAGE
procedure OMSendMessage(var msg: TOmniMessage); message MSG_SEND_MESSAGE;
. . .
end;
В принципе все это мы уже проходили и все это нам понятно. Послали сообщение – обработали его. Однако можно переписать данный код другим способом. Дело в том, что OTL позволяет указывать имя метода или ссылку на этот метод в параметрах. На примере метода SetTimer это выглядит так:
SetTimer
procedure SetTimer(interval_ms: cardinal; const timerMethod: pointer); overload;
procedure SetTimer(interval_ms: cardinal; const timerMessageName: string); overload;
В первом случае Вы можете передавать указатель на метод обработчика данного сообщения, а во втором указать имя метода обработчика, которое должно быть вызвано OTL при срабатывании таймера.
Если мы перепишем задачу, то обработчик в фоновой задаче запишем так:
TAsyncHello = class(TOmniWorker)
public
. . .
// Наш обработчик MSG_SEND_MESSAGE
procedure SendMessage;
end;
Вызов же может быть осуществлен таким способом
Вызов метода фоновой задачи
var
worker: IOmniWorker;
begin
worker := TAsyncHello.Create;
FHelloTask :=
OmniEventMonitor1.Monitor(CreateTask(worker, 'Hello'));
// Вызов по имени метода TAsyncHello
FHelloTask.SetTimer(1000, 'SendMessage')
// Или вызов по адресу метода TAsyncHello.SendMessage
FHelloTask.SetTimer(1000, @TAsyncHello.SendMessage);
FHelloTask.Run;
Если вы используете вызов метода по имени, знайте, что OTL не проверяет достоверность задания имени метода в фоновой задаче. То есть если Вы сделаете опечатку и напишете, например
FHelloTask.SetTimer(1000, 'SendMessages') //последняя s лишняя
то OTL не найдет такой метод и будет сгенерированная ошибка.
Еще важной особенность является то, что при использовании вызова по адресу, Вы не имеете права указывать адрес метода, который не находится в фоновой задаче (потоке класса TOmniWorker). Такой вызов приведет к ошибке:
type
TAsyncHello = class(TOmniWorker)
public
procedure SendMessage;
end;
TfrmTest = class(TForm)
. . .
procedure SendMessage;
. . .
end;
. . .
FHelloTask.SetTimer(1000, @TfrmTest.SendMessage); // так делать нельзя
С эти мы разобрались. До этого весь наш раздел строился на разборе метода SetTimer, что было хорошо для введения. Продолжим далее, библиотека OTL позволяет Вам вызывать любые методы фоновой задачи (а также передавать в них данные) используя для этого метод Invoke интерфейса IOmniTaskControl.
Способы вызова метода Invoke представлены далее:
Invoke
function Invoke(const msgMethod: pointer): IOmniTaskControl; overload;
function Invoke(const msgMethod: pointer; msgData: array of const): IOmniTaskControl; overload;
function Invoke(const msgMethod: pointer; msgData: TOmniValue): IOmniTaskControl; overload;
function Invoke(const msgName: string): IOmniTaskControl; overload;
function Invoke(const msgName: string; msgData: array of const): IOmniTaskControl; overload;
function Invoke(const msgName: string; msgData: TOmniValue): IOmniTaskControl; overload;
Как видите, ничего сложного в этом методе нет. На входе в первом параметре Invoke так-же принимает либо имя метода задачи, либо адрес на метод. Второй параметр служит для передачи некоторых данных. Тип TOmniValue будет более подробно рассмотрен нами в разделе “Внутренняя организация OTL”.
Чтобы закрепить на практике наши знания рассмотрим простой пример:
Вызов метода фоновой задачи из основного потока
// наша фоновая задача
TAsyncHello = class(TOmniWorker)
strict private
aiMessage: string;
public
function Initialize: boolean; override;
published
procedure Change(const data: TOmniValue);
procedure SendMessage;
end;
. . .
// ФОНОВАЯ ЗАДАЧА ЗАПУЩЕНА
// Из GUI-потока мы всегда сможем вызвать любой наш метод с помощью Invoke
// FHelloTask – интерфейс управления IOmniTaskControl нашей фоновой задачи
. . .
procedure TfrmTestStringMsgDispatch.btnChangeMessageClick(Sender: TObject);
begin
// Вызываем метод Change по имени и посылаем случайно сгенерированную строку
FHelloTask.Invoke('Change', 'Random ' + IntToStr(Random(1234)))
// Либо вызываем метод по адресу на него
FHelloTask.Invoke(@TAsyncHello.Change, 'Random ' + IntToStr(Random(1234)));
end;
// Наш метод, который вызван из основного потока
procedure TAsyncHello.Change(const data: TOmniValue);
begin
aiMessage := data;
end;
Как видите, вызвать определенный метод фоновой задачи не представляется сложным. С OTL идет демонстрационный пример tests\18_StringMsgDispatch, который прекрасно демонстрирует использование данного метода. Используя метод Invoke вы легко сможете наделять Ваши фоновые задачи многофункциональностью и гибкостью разделяя разный функциональный код в методах.
Комментариев нет:
Отправить комментарий