2

我正在使用 XE8,并且正在尝试构建我的真实世界应用程序的示例。

我需要在主“服务线程”和 OTL 线程池之间进行通信。这些示例都设置了表单和监视器。我不需要这些,但我想不出一种方法来编写干净的代码。到目前为止,这就是我所做的:

TProcessWorker = Class( TOmniWorker )
strict private
  FTaskID : int64;
  FIndex : Integer;
  FFolder : String;
protected
  function Initialize: Boolean; override;
public
  procedure WriteTask( var msg : TMessage); message _AM_WriteTask;
End;

{ TProcessWorker }

function TProcessWorker.Initialize: Boolean;
begin
  FTaskID := Task.UniqueID;
  FIndex := 0;
  result := True;
  FFolder := Format('%s/%d', [Task.Param['Folder'].AsString, FTaskID]);
  ForceDirectories(FFolder);
end;

实现为:

procedure TProcessWorker.WriteTask(var msg: TMessage);
var
  ps : PString;
  L : TStringStream;
begin
   Ps:= PString(msg.LParam);
   L := TStringStream.Create( ps^ );
   try
     L.SaveToFile( format('%s\%d.txt',[FFolder, fIndex]) );
   finally
     l.Free;
     inc(FIndex);
   end;
end;

在主线程中,为了创建池,我调用:

FThreadPool := CreateThreadPool('Thread pool test');

var
  lFolder : String;
  Process : IOmniWorker;
begin
   lFOlder := ExtractFilePath(ParamStr(0));
   Process := TProcessWorker.Create;
   CreateTask( Process, 'Task test').Unobserved.SetParameter('Folder',lFolder).Schedule(FThreadPool);

我不知道如何正确调用我的工作线程。在我的实际应用程序中,将触发多个线程,我需要确保我正确使用了线程池。

1)通过像我一样调用CreateTask,我如何正确使用线程池?为我需要的每个进程调用 CreateTask 对我来说似乎很奇怪。

2)工作线程永远不会被触发。我应该如何让我的 Worker 线程工作!:)

问候,克莱门特

4

2 回答 2

0

检查http://otl.17slon.com/book/doku.php?id=book:howto:connectionpool

我的感觉是 OTL 基于数据容器,而不是线程。

所以我认为您需要创建一个任务请求队列,您的“主线程”会将任务注入其中。

池的想法是他们自己管理!你不应该与特定的工作线程通信,你应该只向它发送工作请求,然后让池在它认为合适的时候产生/杀死工作线程。

如果您需要来自每个特定线程的反馈,我宁愿将 TForm.Handle 或 TOmniMonitor 指针包含到任务请求记录中,并让工作线程回调并与表单进行通信,而不是来自线程

于 2015-06-08T14:34:10.900 回答
0

OmniThreadLibrary 测试08_RegisterComm展示了如何在两个线程之间直接通信。

基本上,您必须创建一个实例IOmniTwoWayChannel并将其端点注册到 worker 的Initialize方法中,使用Task.RegisterComm(<channel>).

然后,您可以以“正常”方式发送消息,<channel>.Send(<message>, <data>)如果您以 Delphi 方式装饰它,它们将被分派到其他任务的消息方法:

procedure MessageHandler(var msg: TOmniMessage); message <message>;
于 2015-06-19T11:03:15.387 回答