TTask比TThread慢

时间:2017-06-09 01:44:34

标签: multithreading delphi delphi-10.2-tokyo

我有两种多线程代码执行相同工作的方法。

TTask:

const
  MaxThreadCount = 80;

procedure TWorkerTask.Update;
var
  aTasks  : array of ITask;
  I: Integer;
begin
  Stopwatch := TStopwatch.StartNew;
  SetLength(aTasks, MaxThreadCount);
  for I := Low(aTasks) to High(aTasks) do begin
    aTasks[I] := TTask.Create( procedure
    begin
      Writeln('Thread ', TTask.CurrentTask.Id, ' Stared');
      Sleep(5000);
      Writeln('Thread ', TTask.CurrentTask.Id, ' Finshed');
    end);
    aTasks[I].Start;
  end;
  TTask.WaitForAll(aTasks);
  Elapsed := Stopwatch.Elapsed;
  Writeln('Done in ', Round(Elapsed.TotalSeconds));
end;

输出,例如Done in 29

的TThread:

const
  MaxThreadCount = 80;

procedure TWorker.Execute;
begin
  Writeln('Thread ', ThreadID, ' Stared');
  Sleep(5000);
  Writeln('Thread ', ThreadID, ' Finshed');
end;

....

var
  Workers   : array of TWorker;
  I         : Integer;
  Stopwatch : TStopwatch;
  Elapsed   : TTimeSpan;  
begin
  SetLength(Workers, MaxThreadCount);
  for I := Low(Workers) to High(Workers) do begin
    Workers[I] := TWorker.Create;
    Workers[I].Start;
  end;
  for I := Low(Workers) to High(Workers) do
    Workers[I].WaitFor;
  Elapsed := Stopwatch.Elapsed;
  Writeln('Done ', Round(Elapsed.TotalSeconds));

输出,例如Done 8

问:为什么TTask比上述方法中的TThread类慢得多?有没有办法加快速度以获得类似的结果?

2 个答案:

答案 0 :(得分:8)

这是因为你的线程和任务不起作用。

您拥有的线程多于处理器。在使用线程的版本中,您可以为每个任务创建一个线程,虽然处理器是超额订阅的,但这并不重要,因为线程处于休眠状态。

在基于任务的版本中,通常每个处理器有一个线程。所以并非所有任务都可以同时运行。

如果您在充分利用CPU运行繁忙的工作时更换睡眠,那么您会看到两个版本的执行方式相似。事实上,我希望基于任务的版本更好,因为它不会超额订阅处理器。

答案 1 :(得分:0)

每个任务不一定意味着1个线程。 delphi任务系统可以节省资源,甚至可以监视CPU使用率来决定是否创建更多线程。特别是我也发现任务很慢,并且为了避免这种缓慢,当我需要使用任务时,我只是为每个任务创建一个线程池,因此它的性能与简单的TThread相同,因为这样您可以确保每个任务都可以在自己的线程上运行。

您可以尝试以下操作:

TTask.Run(
  procedure()
  begin
  end, TThreadPool.Create);