如何使用具有两个不同类但其中一个来自 VCL 的泛型

Posted

技术标签:

【中文标题】如何使用具有两个不同类但其中一个来自 VCL 的泛型【英文标题】:How to use a generic with two different classes but one of them from the VCL 【发布时间】:2018-11-24 23:22:14 【问题描述】:

我正在维护和扩展一个使用 8 位位图的旧版应用程序,但出于多种原因,我从头开始创建自己的位图类 (TMyBitmap),实现了一些明显的功能,例如 Width、Height 和 ScanLine 属性。请注意,TBitmap 和 TMyBitmap 之间没有关系(TMyBitmap 不继承自 TBitmap)。

现在我想做一个实用类,它有几种方法可以使用标准 TBitmap 或我自己的 TMyBitmap。这些方法的实现对于 TBitmap 或 TMyBitmap 是相同的,因为它们只使用“通用”属性(Width、Height 和 ScanLine),因此泛型看起来是一个合适的解决方案:

TBmpUtility<T> = class
strict private
  FBmp: T;
public
  constructor Create(Bmp: T);
  procedure Fill(Y: Integer; Code: Byte); //example of an utility method
end;

procedure TBmpUtility<T>.Fill(Y: Integer; Code: Byte);
var
  i: Integer;
  Line: PByteArray;
begin
  Line := Self.FBmp.ScanLine[Y];
  for i := 0 to Self.FBmp.Width - 1 do
    Line[i] := Code;
end;

这里明显的问题是我不能真正调用Self.FBmp.ScanLine[Y]Self.FBmp.Width,因为此时编译器对 TBitmap 或 TMyBitmap 一无所知。然后我声明了一个接口并在 TMyBitmap 上实现:

  IBitmap  = interface
  ['029D42A0-132F-4C6E-BE72-648E1361C0A4']
    function GetWidth: Integer;
    procedure SetWidth(Value: Integer);
    function GetHeight: Integer;
    procedure SetHeight(Value: Integer);
    function GetScanLine(Y: Integer): Pointer;
    property Width: Integer read GetWidth write SetWidth;
    property Height: Integer read GetHeight write SetHeight;
    property ScanLine[Y: Integer]: Pointer read GetScanLine;
  end;

  TMyBitmap = class (TSinglentonImplementation, IBitmap)
  ...
  end;

然后我可以在泛型声明上添加类型约束,以便能够从实用方法实现中通过Self.FBmp 调用这些方法。

TBmpUtility<T: IBitmap> = class

这里的问题是:标准的 TBitmap 没有实现我的 IBitmap 接口。所以我不能使用带有标准位图的实用程序类,如下所示:

Bmp := TBitmap.Create();
BmpUtil := TBmpUtility<TBitmap>.Create(Bmp);

我可以这样做:

TCustomBitmap = class (TBitmap, IBitmap)
...
end;

但是项目代码到处都在使用 TBitmap,而不是 TCustomBitmap,所以我不能互换这些类型,所以这不是一个选项。

我尝试了一个辅助类。像这样的:

TBitmapHelper = class (IBitmap) helper for TBitmap
...
end;

但显然,它不起作用

知道如何解决这种情况吗? 记住一开始的目标:只有一个实现代码块用于处理 TBitmap 和 TMyBitmap 的实用程序方法。如果泛型是正确的解决方案,它应该如下所示:

Bmp := TBitmap.Create();
BmpUtil := TBmpUtility<TBitmap>.Create(Bmp);
BmpUtil.Fill(10, 0);
//or
MyBmp := TMyBitmap.Create();
BmpUtil := TBmpUtility<TMyBitmap>.Create(MyBmp);
BmpUtil.Fill(10, 0);

(仅供参考:我使用的是 Delphi 10.2.3 [东京])

【问题讨论】:

我不认为泛型在这里有用。您需要通过将调用转发到 TBitmap 实例来创建一个实现接口的适配器类。 相关:Invoke method on generic type? 【参考方案1】:

正如 David 所建议的,您可以创建一个适配器类。此类可以实现您的接口并(在大多数情况下)作为对实际位图类的简单传递。用于 TBitmap 的可能如下所示(为简洁起见,省略了一些方法):

type
  TBitmapAdapter = class(TInterfacedObject, IBitmap)
  private
    FBitmap: TBitmap;
    function GetWidth: Integer;
    procedure SetWidth(Value: Integer);
    // And everything else from IBitmap
  public
    constructor Create(ABitmap: TBitmap);
  end;

function TBitmapAdapter.GetWidth: Integer;
begin
  Result := FBitmap.Width;
end;

procedure TBitmapAdapter.SetWidth(Value: Integer);
begin
  FBitmap.Width := Value;
end;

您可以创建一个类来处理这两种类型,但这需要在每个方法中使用 if。我会创建两个单独的类,因此您可以简单地创建一个 TBitmapAdapter 或 TMyBitmapAdapter。

您可以简单地构造该类并将其作为接口传递。如果你的 TBmpUtility 可以接受一个 IBitmap,你可以这样构造它:

TBmpUtility.Create(TBitmapAdapter.Create(Self))

您确实可以为 TBitmap 和 TMyBitmap 编写类助手来为您创建正确的包装器。还是 TBitmap 版本:

type
  TBitmapHelper = class helper for TBitmap
    function AsIBitmap: IBitmap;
  end;

function TBitmapHelper.AsIBitmap: IBitmap;
begin
  Result := TBitmapAdapter.Create(Self);
end;

然后您可以像这样简单地使用它:

TBmpUtility.Create(Bitmap.AsIBitmap)

或者也许只是将它传递给实用程序类的方法,您可以拥有一个可以处理任何位图的实用程序实例:

BmpUtility.Fill(Bitmap.AsIBitmap);

可以临时存储对 IBitmap 的引用,以防止在每次调用时创建和销毁适配器:

var b: IBitmap;
begin
  b := Bitmap.AsIBitmap;
  BmpUtility.SetHeight(b, 100);
  BmpUtility.SetWidth(b, 100);
  BmpUtility.Fill(b);
end;

Bitmap.AsIBitmap的结果是适配器类的IBitmap接口,但是使用它的代码甚至都不知道,并且接口是refcount的,所以一旦超出范围就会释放适配器.

【讨论】:

以上是关于如何使用具有两个不同类但其中一个来自 VCL 的泛型的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Java 中使用来自不同类的方法?

如何使来自不同类的文本字段等于相同的文本?

GetMethod 的 Type[] 以获取具有两个参数和一个类型参数的泛型方法

Cocoa:将子视图添加到来自不同类和 nib 的视图

如何将小部件推广到具有不同选择的不同类?

如何将不同类中的两个 Switch 控件绑定到同一个 bool 属性?