14.2.4 泛型类型函数
到目前为止,我们遇到的泛型类型定义的最大问题是,您对泛型类型元素能做的事情非常少。有两种技术可以用来克服这个限制。第一种是利用运行时库的一些特殊函数,这些函数专门支持泛型类型;第二种(也是更强大的一种)是在定义泛型类时对可以使用的类型进行限制。
本节将重点介绍第一种方法,下一节将重点介绍约束。正如我提到的,有一些 RTL 函数可以处理泛型定义的参数类型 (T):
-
Default(T)
实际上是与泛型一起引入的新函数,它返回当前类型的空值或“零值”或null值,可以是零、空字符串、nil等;事实上,全局变量与局部变量不同,编译器会将全局变量初始化为 “零”) -
TypeInfo(T)
返回指向当前版本的泛型类型的运行时信息的指针;有关类型信息的更多信息,请参见第16章 -
SizeOf(T)
返回类型的字节数(对于引用类型,如字符串或对象,它将是引用的大小,即32位编译器的4个字节,64位编译器的8个字节) -
IsManagedType(T)
指示类型是否在内存中受管理,例如字符串和动态数组 -
HasWeakRef(T)
与支持ARC的编译器相关,表示目标类型是否具有弱引用;这需要特定的内存管理支持 -
GetTypeKind(T)
是一种从类型信息访问类型种类的快捷方式;它是比TypeInfo
返回的类型定义稍高一级的类型定义
注意 所有这些方法都返回编译器评估过的常量,而不是在运行时调用实际函数。这一点的重要性并不在于这些操作非常快,而在于这使编译器和链接器可以优化生成的代码,删除未使用的分支。如果有一个 case 或 if 语句是基于这些函数之一的返回值,编译器立即就能算出,仅对于给定类型,其中一个分支将被执行,从而删除无用的代码。当针对不同类型编译相同的泛型方法时,可能会使用不同的分支,但编译器同样可以预先计算并优化方法的大小。
GenericTypeFunc
示例中有一个泛型类,展示了三个泛型类型的函数的作用:
typeTSampleClass<T> = classprivateFData: T;publicprocedure Zero;function GetDataSize: Integer;function GetDataName: string;end;function TSampleClass<T>.GetDataSize: Integer;beginResult := SizeOf(T);end;function TSampleClass<T>.GetDataName: string;beginResult := GetTypeName(TypeInfo(T));end;procedure TSampleClass<T>.Zero;beginFData := Default(T);end;
在GetDataName
方法中,我使用了GetTypeName
函数(System.TypInfo
单元的函数),而不是直接访问数据结构,因为它可以执行保持类型名称的字符串值编码的正确转换。
给定上述声明,您可以编译以下测试代码,该代码在三个不同的泛型类型实例上重复三次。我省略了重复的代码,并仅显示用于访问数据字段的语句,因为它们会根据实际类型而改变:
varT1: TSampleClass<Integer>;T2: TSampleClass<string>;T3: TSampleClass<Double>;
beginT1 := TSampleClass<Integer>.Create;T1.Zero;Show('TSampleClass<Integer>');Show('Data: ' + IntToStr(T1.FData));Show('Type: ' + T1.GetDataName);Show('Size: ' + IntToStr(T1.GetDataSize));T2 := TSampleClass<string>.Create;Show('Data: ' + T2.FData);T3 := TSampleClass<Double>.Create;Show('Data: ' + FloatToStr(T3.FData));
运行此代码(来自GenericTypeFunc示例)会产生以下输出:
TSampleClass<Integer>
Marco Cantù, Object Pascal Handbook 11
396 - 14: generics
Data: 0
Type: Integer
Size: 4
TSampleClass<string>
Data:
Type: string
Size: 4
TSampleClass<Double>
Data: 0
Type: Double
Size: 8
请注意,您也可以在泛型类的上下文之外,对特定类型使用泛型类型函数。例如,您可以编写:
varI: Integer;s: string;
beginI := Default(Integer);Show('Default Integer: ' + IntToStr(I));s := Default(string);Show('Default String: ' + s);Show('TypeInfo String: ' + GetTypeName(TypeInfo(string));
这是一个简单的输出:
Default Integer: 0
Default String:
TypeInfo String: string
注解:您不能像上面的代码中的
TypeInfo(s)
那样将TypeInfo
调用应用于变量,而只能将其应用于数据类型。