gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-03 18:30:01 24 4
gpt4 key购买 nike

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

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

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 一无所知。然后我声明了一个接口(interface)并在 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 接口(interface)。所以我不能使用带有标准位图的实用程序类,如下所示:
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 [东京])

最佳答案

正如大卫建议的那样,您可以制作一个适配器类。此类可以实现您的接口(interface)并(在大多数情况下)作为对实际位图类的简单传递。用于 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。

您可以简单地构造该类并将其作为接口(interface)传递。如果您的 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 接口(interface),但是使用它的代码甚至都不知道,并且接口(interface)被引用,所以适配器一旦超出范围就会被释放。

关于delphi - 如何使用具有两个不同类但其中一个来自 VCL 的泛型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53463253/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com