首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >对象池-同步- Delphi

对象池-同步- Delphi
EN

Stack Overflow用户
提问于 2012-08-30 22:45:26
回答 3查看 1.6K关注 0票数 1

我正在用Delphi实现一个对象池。我需要同步线程,以便从池中获取对象。

线程代码:

代码语言:javascript
复制
uClientQueryPool.CLIENT_POOL_GUARD.Acquire();
QueryClient := QUERY_POOL.GetClient();
uClientQueryPool.CLIENT_POOL_GUARD.Release;

池代码:

代码语言:javascript
复制
var
   CLIENT_POOL_GUARD: TCriticalSection;

type
   TClientQueryPool = class
public
   function GetClient(): TQueryClient;
end;

CLIENT_POOL_GUARD是一个单位变量。池工作得很好,但是我可以在GetClient方法中使用"uClientQueryPool.CLIENT_POOL_GUARD.Acquire();“和"uClientQueryPool.CLIENT_POOL_GUARD.Release;”吗?

如下所示:

代码语言:javascript
复制
function TClientQueryPool.GetClient: TQueryClient;
begin
    CLIENT_POOL_GUARD.Acquire();
    ...
    CLIENT_POOL_GUARD.Release;
end;
EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2012-08-31 00:45:07

将锁移动到get/pop/任何方法中就可以了,就像让CriticalSection实例成为pool类的私有成员一样。在release()调用中使用相同的CS,将对象推回到池中。

几十年来,我一直在这样做,通常使用TObjectQueue作为池队列,使用CS来保护它,使用信号量来计算池内容,以及在池暂时清空时请求线程阻塞的东西。

不知道那个‘双重获取’的帖子是从哪里来的。锁要么在池类内部,要么在外部。我真的无法想象为什么会有人同时编写这两个代码!

示例类:

首先,线程安全的P-C队列,用于保存池化对象:

代码语言:javascript
复制
unit tinySemaphoreQueue;

interface

uses
  Windows, Messages, SysUtils, Classes,syncObjs,contnrs;


type

pObject=^Tobject;


TsemaphoreMailbox=class(TobjectQueue)
private
  countSema:Thandle;
protected
  access:TcriticalSection;
public
  property semaHandle:Thandle read countSema;
  constructor create; virtual;
  procedure push(aObject:Tobject); virtual;
  function pop(pResObject:pObject;timeout:DWORD):boolean;  virtual;
end;


implementation

{ TsemaphoreMailbox }

constructor TsemaphoreMailbox.create;
begin
  inherited Create;
  access:=TcriticalSection.create;
  countSema:=createSemaphore(nil,0,maxInt,nil);
end;

function TsemaphoreMailbox.pop(pResObject: pObject;
  timeout: DWORD): boolean;
begin // wait for a unit from the semaphore
  result:=(WAIT_OBJECT_0=waitForSingleObject(countSema,timeout));
  if result then // if a unit was supplied before the timeout,
  begin
    access.acquire;
    try
      pResObject^:=inherited pop; // get an object from the queue
    finally
      access.release;
    end;
  end;
end;

procedure TsemaphoreMailbox.push(aObject: Tobject);
begin
  access.acquire;
  try
    inherited push(aObject); // shove the object onto the queue
  finally
    access.release;
  end;
  releaseSemaphore(countSema,1,nil); // release one unit to semaphore
end;

end.

然后是对象池:

代码语言:javascript
复制
unit tinyObjectPool;

interface

uses
  Windows, Messages, SysUtils, Classes,syncObjs,contnrs,
  tinySemaphoreQueue;

type
  TobjectPool=class;

  TpooledObject=class(TObject)
  private
    FmyPool:TObjectPool;
  protected
    Fparameter:TObject;
  public
    procedure release;
    constructor create(parameter:TObject); virtual;
  end;

  TpooledObjectClass=class of TpooledObject;

  TobjectPool=class(TsemaphoreMailbox)
  private
    Fparameter:TObject;
    function getPoolLevel: integer;
  public
    property poolLevel:integer read getPoolLevel;
    constructor create(poolDepth:integer;
      pooledObjectClass:TpooledObjectClass;parameter:TObject); reintroduce; virtual;
  end;

implementation

{ TobjectPool }

constructor TobjectPool.create(poolDepth: integer;
  pooledObjectClass: TpooledObjectClass;parameter:TObject);
var objectCount:integer;
    thisObject:TpooledObject;
begin
  inherited create;
  Fparameter:=parameter; // a user parameter passed to all objects
  for objectCount:=0 to poolDepth-1 do // fill up the pool with objects
  begin
    thisObject:=pooledObjectClass.create(parameter);
    thisObject.FmyPool:=self;
    inherited push(thisObject);
  end;
end;

function TobjectPool.getPoolLevel: integer;
begin
  access.acquire;
  result:=inherited count;
  access.release;
end;



{ TpooledObject }

constructor TpooledObject.create(parameter: TObject);
begin
  inherited create;
  Fparameter:=parameter;
end;

procedure TpooledObject.release;
begin
  FmyPool.push(self);
end;

end.
票数 2
EN

Stack Overflow用户

发布于 2012-08-31 01:38:04

是的你可以。请注意,尽管您可以以线程安全的方式从池中提取对象,但如果对象本身不是线程安全的,则使用它可能不是线程安全的。例如,在下面的示例中,池是线程安全的,甚至在池中的所有对象都在使用的情况下使线程等待,但一旦对象正在使用,使用它仍然不是线程安全的,因为它使用全局数据。

代码语言:javascript
复制
uses
  SyncObjs;

var
  GlobalData: Integer = 0;

type
  TDataObject = class
    Used: Boolean;
    procedure UpdateData;
  end;

type
  TPool = class
    FLock: TCriticalSection;
    FSemaphore: TSemaphore;
    FDataObjects: array[0..9] of TDataObject;
    constructor Create;
    destructor Destroy; override;
    function GetDataObject: TDataObject;
    procedure ReleaseDataObject(AObject: TDataObject);
  end;

var
  Pool: TPool;

type
  TDataThread = class(TThread)
    constructor Create;
    procedure Execute; override;
  end;

{ TPool }

constructor TPool.Create;
var
  i: Integer;
begin
  inherited Create;
  FLock := TCriticalSection.Create;
  FSemaphore := TSemaphore.Create(nil, Length(FDataObjects), Length(FDataObjects), '', False);

  for i := Low(FDataObjects) to High(FDataObjects) do
    FDataObjects[i] := TDataObject.Create;
end;

destructor TPool.Destroy;
var
  i: Integer;
begin
  for i := Low(FDataObjects) to High(FDataObjects) do
    FDataObjects[i].Free;

  FSemaphore.Free;
  FLock.Free;
end;

function TPool.GetDataObject: TDataObject;
var
  i: Integer;
begin
  Result := nil;

  FLock.Acquire;
  try
    FSemaphore.Acquire;
    for i := Low(FDataObjects) to High(FDataObjects) do
      if not FDataObjects[i].Used then
      begin
        Result := FDataObjects[i];
        Result.Used := True;
        Exit;
      end;

    Assert(Result <> nil, 'Pool did not return an object');
  finally
    FLock.Release;
  end;
end;

procedure TPool.ReleaseDataObject(AObject: TDataObject);
begin
  if not AObject.Used then
    raise Exception.Create('Data object cannot be released, because it is not in use.');

  AObject.Used := False;
  FSemaphore.Release;
end;

{ TDataObject }

procedure TDataObject.UpdateData;
begin
  Inc(GlobalData);
end;

procedure TForm1.Button1Click(Sender: TObject);
begin
  TDataThread.Create;
end;

{ TDataThread }

constructor TDataThread.Create;
begin
  inherited Create(True);
  FreeOnTerminate := True;
  Resume;
end;

procedure TDataThread.Execute;
var
  DataObject: TDataObject;
begin
  DataObject := Pool.GetDataObject;

  DataObject.UpdateData; // <-- Not thread-safe!

  Pool.ReleaseDataObject(DataObject);
end;

initialization
  Pool := TPool.Create;
finalization
  Pool.Free;
end.
票数 2
EN

Stack Overflow用户

发布于 2012-08-30 23:08:17

1)我会从线程代码中删除获取/释放代码-它是脆弱的。在一个线程中,你忘记了调用它,然后砰!根据经验,安全措施应该由服务器集中实施,而不是以模糊的方式分布在客户端。

2)获取/释放调用应该防止出现错误,否则任何错误异常都将永远锁定所有线程。

代码语言:javascript
复制
 function TClientQueryPool.GetClient: TQueryClient;
 begin
   CS.Acquire;
   try
     // actually getting object, preferably just calling
     // internal non-public thread-unsafe method for it
   finally
     CS.Release;
   end;
  end;

3)临界区本身最好是池的内部非公共成员。这样一来,当你忘记了实现细节时,你就可以轻松重构了,比如:

3.1)实现多个池

3.2)将池代码移动到另一个单元

3.3)确保池外的任何错误代码都不能通过随机获取或释放CS来使应用程序崩溃

4)在TCriticalSection对象上双重调用acquire/release,将所有赌注放在TCriticalSection文档中The_Fox所指向的单个注释的含义上。每次对Release的调用都应该与之前获取“http://docwiki.embarcadero.com/Libraries/en/System.SyncObjs.TCriticalSection.Release”的调用相平衡

并希望今天和明天所有其他的Pascal实现都不会错过它。

这是一种脆弱的实践。多线程代码以创建Heisenbug而闻名,当客户端站点出现问题时,但您无法在内部重现和找到它。如果您的公司将来扩展到不同的平台或不同的语言实现,这将是一个潜在的地雷。而这种类型的矿藏,很难通过室内测试找到。多线程代码是你最好是过度失败的地方,只要不允许任何不确定性发生。

票数 -3
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/12199547

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档