現在、ジェネリック クラスのクラス型を宣言することはできません。
詳細については、 QC76605を参照してください。また、以下の更新。
例 :
TMyClass<T> = class
end;
TMyClassClass<T> = class of TMyClass<T>; //E2508 type parameters not allowed on this type
提示される回避策は次のようになります。
TMyIntClass = TMyType<Integer>;
TMyIntClassClass = Class of TMyIntClass;
しかし、コメントしたように、ジェネリックのインスタンス化ごとにクラスをサブクラス化する必要があるため、ジェネリックのアイデア全体が無効になります。
ジェネリック型の特殊化されたサブクラスを生成するための同様の回避策へのリンクもここにあります: destroy-from-specialized-generic-types。この場合、次のようになります。
TMySpecialClass = Class(TMyType<Integer>);
アップデート :
RM によって提案された回避策:
TMyType<T> = class(a.TMyType<T>);
次のスキームを使用して、タイプセーフで実装できます。
unit Unita;
interface
type
TMyType<T> = class
Constructor Create;
end;
implementation
uses
Unitb;
constructor TMyType<T>.Create;
begin
Inherited Create;
//WriteLn( Self.QualifiedClassName,' ',Unitb.TMyType<T>.QualifiedClassName);
Assert(Self.QualifiedClassName = Unitb.TMyType<T>.QualifiedClassName);
end;
end.
unit Unitb;
interface
uses Unita;
type
TMyType<T> = class(Unita.TMyType<T>);
implementation
end.
Project Test;
{$APPTYPE CONSOLE}
uses
System.SysUtils,
Unita in 'Unita.pas',
Unitb in 'Unitb.pas';
var
t1 : Unita.TMyType<Integer>;
t2 : Unitb.TMyType<Integer>;
t3 : TMyType<Integer>;
begin
try
//t1 := Unita.TMyType<Integer>.Create; //Exception EAssertionFailed !!
t2 := Unitb.TMyType<Integer>.Create;
t3 := TMyType<Integer>.Create;
ReadLn;
finally
//t1.Free;
t2.Free;
t3.Free;
end;
end.
ジェネリック クラスを作成するときに、作成されたクラスがユニット b で宣言された型から派生していることを確認するテストが行われます。これにより、ユニット a からこのクラスを作成しようとするすべての試みが検出されます。
更新 2:
明確にするために、ジェネリック クラス " class of type<T>
" への参照はできませんが、ジェネリック クラスのコピーは問題ありません。