delphi D11编程语言手册 学习笔记(P344-392) 接口/类操作

news2025/1/19 3:13:34

P344-365 接口

  "接口" 的概念和 "类" 特别是 "抽象类" 近似, Delphi 之初并没有接口, 后来(Delphi 3)为了支持 COM 引入了接口, 再后来发展成为 Delphi 重要的语言特性.

  使用 COM 步骤可能是这样的:
    1.程序在使用组件之初, 先联系 "接口"; 这应该是从注册表中查询.
    2.找到后, 如果此接口还没有被实现, 马上调用 "类厂" 建立起对象, 并同时给接口计数器加 1.
    3.可能会不止一个程序在使用同一个接口, 但每有使用则 "计数器+1", 每次用完则 "计数器-1".
    4.当接口的使用计数为 0 时, 系统自动释放类厂为该接口建立的对象(垃圾回收机制).

下面用代码来理解这段话:

Type 
IFace = interface  //声明一个接口
    procedure proc;
  end;

  TAType = class(TInterfacedObject, IFace)  //调用接口
   constructor Create;  //构造函数
    destructor Destroy; override;    //解析函数
   procedure proc;  //在存在构造函数或解析函数时,方法属性一律写在最后,否则出错
  end;

var
  Form2: TForm2;

implementation

{$R *.dfm}

{ TAType }

constructor TAType.Create;
begin
  inherited;
  Form2.memo1.lines.add('TAType  Constructor!');
end;

destructor TAType.Destroy;
begin
  inherited;
  Form2.memo1.lines.add('TAType  Destroy!');
end;

procedure TAType.proc;
begin
  Form2.memo1.lines.add('TAType.proc!');
end;

procedure TForm2.Button1Click(Sender: TObject);
var
  A: IFace;    //注意这里,这里声明的是一个接口
begin
   A:= TAType.Create ;    //然而这里去是用的类的构造函数,这里是接口的一个妙用.接口变量可以接一个实现了该接口的类 
   A.proc;
  //A:=nil;    //主动释放内存,同样的,拥有它的类也会被释放掉!
   Form2.memo1.lines.add('---end---');
end;

  与类相比,接口侧重于封装,并提供与类之间一种比继承更宽松一点的连接.

除了宣告抽象类(拥有抽象方法的类别),在 Object Pascal 里面我们也可以撰写纯粹的抽象类;也就是只包含虚拟抽象方法的类别。透过使用特别的关键词,interface 来定义一组作为接口(interfaces)的数据型别。

从技术面来看,接口不算是类,虽然接口可以重组类.因为类可以建立实体,但是接口不行.接口可以被一个或者多个类调用,所以这些实体就可以算是支持了或者调用了该接口.讲人话就是: 接口是用来给类调用的抽象类.

类与接口的对比:

1.  接口类型的变量会修改接口计数器,跟类类型的变量不同,接口提供了一系列的自动内存管理机制

2.  一个类可以继承多个接口(要列在基础类后面):

  TMyClass = class(父类, 接口1, 接口2, ...) 
      //Some Code
  end;

  但是一个接口只能从另一个接口继承过来,而不能同时继承多个接口,如果继承的对象是根接口Iinterface,则可省略.

IMyInterface1 = interface(Iinterface)
    function Func1: Integer;
    function Func2: Integer;
  end;

3.  所有的类都是从TObject中衍生而来,所有的接口都是从IInteface衍生出来的,两都是各自独立正交的架构.

4.  不成文约定:所有类名,除异常类型外以E开头外,其他都以T开头,而所有接口都以字母I开头.

5.  接口只有方法与属性没有字段

6.  接口成员都是公开的,不需要private/protected/public/published等修饰语

7.  因为接口只声明、无实现, 所以也用不到继承与覆盖相关的修饰(virtual、dynamic、abstract、override).

8.  不管实现接口的类有多么丰富, 接口只拥有自己声明的成员.

9.  实现接口的类一般继承于 TInterfacedObject, 直接从 TObject 继承会增加一些麻烦而重复的工作.

10.  接口在用完后会自动释放, 并同时释放拥有它的类; 这很方便, 但同时带来很多问题.

11. 每个接口具有唯一的标识符,称为GUID,在接口内按下Ctrl+Shift+G,IDE会自动产生一组GUID  

 IFace = interface
  //GUID,平常是由系统分配的,所以基本不用理会.这里只是演示
    ['{E5AC8E60-2DBE-43C5-B679-27731A56F3D4}']  
    procedure proc;
  end;

接口的声明与调用:

  和类的声明一样,接口也是在interface区域下用关键字type进行声明:

unit Unit1;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants,
  System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs,
  Vcl.StdCtrls;

type
  TForm1 = class(TForm)
    Memo1: TMemo;
    Button1: TButton;
    procedure Button1Click(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;
//接口只有属性和方法,没有字段!并且只能继承一个祖先,如果是根接口,可以省略不写
 //接口1   
  IFace1 = interface
    function Fn1(Str: string): string; //抽象方法,只有声明没有实体,按Ctrl+Shift+c无响应
  end;
//接口2
  IFace2 = interface
    procedure Proc2;    //抽象方法,只有声明没有实体,按Ctrl+Shift+c无响应
  end;

//类调用上面的两个接口,注意先后顺序,先类,后接口,不然报错.
//接口里有什么方法,调用的类里面就要定义什么方法,说白了就是抄一份下来
//否则会提示 接口方法 IFace1.test 没有实现部分
  TAType = class(TInterfacedObject, IFace1, IFace2)  //先类后接口
   function Fn1(Str: string): string;//这里要按Ctrl+Shift+c来完成方法实体
    procedure Proc2;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

{ TAType }
//方法实体
function TAType.Fn1(str: string): string;
begin
  Result := ('Fn1:' + str);
end;
//方法实体
procedure TAType.Proc2;
begin
  ShowMessage('Fn2');
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  TA: TAType;
begin
  Memo1.Clear;
  Memo1.Lines.Add(TA.Fn1('哈哈哈'));//接口调用
  TA.Proc2;//接口调用
end;

end.

 接口属性:

{此接口声明了一个 Name 属性; 因为接口没有字段, read/write 都只能从方法}
  IMyInterface = interface
    function GetName : string;
    procedure SetName(val : string);
    property Name : string read GetName write SetName;
  end;

  {类实现的是接口的读写方法, 属性还是属于接口的; 类可以提供一个储存属性的字段}
  TMyClass = class(TInterfacedObject, IMyInterface)
  private
    FName: string;
  public
    function GetName: string;
    procedure SetName(val: string);
  end;

 接口可以被多次调用.因为接口的方法是抽象的,实体部分要到调用接口的类里面才具现,所以类调用接口的方法时,方法能执行什么功能,还不是你说的算?

type
  //接口
  IMyInterface1 = interface
    function Func(a,b: Integer): Integer;
  end;
//调用接口的类1,这里的方法执行的是加法
  TAdd = class(TInterfacedObject, IMyInterface1)
  public
    function Func(a: Integer; b: Integer): Integer;  //Result:= a+b;
    destructor Destroy; override;
  end;
//调用接口的类2,这里的方法执行的是乘法
  TMul = class(TInterfacedObject, IMyInterface1)
  public
    function Func(a: Integer; b: Integer): Integer;  //Result:= a*b;
    destructor Destroy; override;
  end;

  弱化引用[weak]使变量的接口计数器不被修改.

type
  TForm1 = class(TForm)
    Memo1: TMemo;
    Button1: TButton;
    procedure Button1Click(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;

  IFace = interface
    procedure test;
  end;

  Ttype = class(TInterfacedObject, IFace)
    destructor Destroy; override;
    procedure test;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

{ Ttype }

destructor Ttype.Destroy;
begin
  Form1.Memo1.Lines.Add('Destroy');  //在实体中调用控件的话,要加上完整的对象位置.
  inherited;
end;

procedure Ttype.test;
begin
  Form1.Memo1.Lines.Add('test');
end;

调用:

procedure TForm1.Button1Click(Sender: TObject);
var
//  [weak]
  a: IFace;
begin
   a:=  Ttype.Create ;
   a.test ;
   Form1.Memo1.Lines.Add('---')
end;

 可以看到,在不使用 weak 弱化引用时,程序正常执行.

 如果把 [weak] 解除注释,则会出现内存错误.因为当a被创建时,因为weak的原因,a的接口计数器没有被加1,仍然等于0,所以系统就自动执行了destroy解析函数,把a给翻译掉了.当再执行a.test时,因为对象已经不存在,所以就会内存报错

 [unSafe]和[weak]不同,它不会主动去清除接口计数器,它是憜性的,下面的例子还是延用上面的代码,可以看到Destroy程序并没有被执行

再看一个网上的例子:

type
  TA=class(TInterfacedObject)
 
  end;
 
procedure TForm1.Button1Click(Sender: TObject);
var
  a:IInterface;
  [weak]aweak:IInterface;
begin
  a:=TA.Create;   //创建对象,复制给a,执行完成后引用计数器=1
  aweak:=a;       //由于aweak定义有[weak]属性,所以赋值给aweak后,引用计数器依旧为1,但aweak变量的地址被保存到一个weak关联列表中
  Memo1.Lines.Add(Format('Ptr:%d', [NativeInt(Pointer(aweak))]));
  a:=nil;         //由于引用计数器=1,执行此句后,计数器清0,对象被释放,同时与此对weak关联列表中所有变量也被赋值为nil,包括aweak变量.
  Memo1.Lines.Add(Format('Ptr:%d', [NativeInt(Pointer(aweak))]));
end;

 运行结果:
Ptr:16360080
Ptr:0
weak引用非常适合用于两个对象需要互相引用的情况下,如果以往的引用,将无法让引用计数器清0.

 方法别名:

  IFace = interface
    ['{3BA60C4E-CA74-489C-B74B-3FA791BC2843}']
    procedure Ptest;
    function Ftest: string;
  end;

  IFace2 = interface
  ['{324AB107-95F1-4145-B392-2E6F555822E3}']
    function Ftest: string;
  end;

  Ttype = class(TInterfacedObject, IFace, IFace2)
    destructor Destroy; override;
    procedure Ptest;
    function F1: string;
    function F2: string;
    function IFace.Ftest = F1;
    function IFace2.Ftest = F2;
  end;

但是我在调用时,调不出F1和F2,不知道问题在哪里,在两个群问了半天,也没有等来答案,如果有大佬知道答案,麻烦指点一下,谢谢啦

 

 P366-392 类操作

  1.类方法与类函数.

    在类里面声明的过程或者函数,只要在前面加上关键字class,就会变成类方法或者类函数.     

    类方法不能在 private 和 protected 区;  

    类方法不能是虚方法;

    类方法只能使用类中的、在对象实例化以前的数据.

    类方法中包含一个隐藏参数self,它是指向类本身的一个参数

    字段不能定义在published区域.在没有任何修饰符的情况下,默认就是published.

  Ttest = class
  private
    class var
      i: Integer;
      var
      n: Integer;
  public
    var
      j: Integer;
    class var
      k: Integer;
    class procedure Ptest;
    class function Ftest(o: integer): Integer;
  end;
---------------------------------
{ Ttest }

class function Ttest.Ftest(o: integer): Integer;
begin
  Result := o * 5;
end;

class procedure Ttest.Ptest;
begin
  i := k * 2;
  ShowMessage(i.ToString);
 ShowMessage(Self.ClassName);
end;
//调用
procedure TForm1.Button1Click(Sender: TObject);
begin
  Ttest.i := 100;    //i=100
//  Ttest.n := 100;  //ERROR,需要create
//  Ttest.j := 100;   // ERROR,需要create
  Ttest.k := 100;   //k=100
  Ttest.Ptest;     //show 200(K*2) show Ttest
  ShowMessage(ttest.Ftest(10).ToString);//show 50(10*5) 
end;

上面的代码中,从头到尾都没有对TTest进行create过,但是对于在定义阶段有加class的方法和字段,跟全局变量一样都是可以直接拿来使用的.相反,没加class的字段,在直接使用时全部都出现了异常.因为他们都需要创建实例后才能正常使用.

警告:使用没有经过初始化(赋值)的字段会抛出异常错误!

类方法就是通过类名就可以访问的方法.或者你可以把它理解为命名空间也行.因为全局变量不被提倡,放到类里面就很好处理了,因为类里面有private,  strict private ,public ,published ,protected等各种保护限制.当你要调用这些类方法或者类函数时,你得这么写: 类名.类方法,或者 类名 .类函数,这样就达到命名空间的效果.

调用时,这个类可以完全不用create而直接使用它内部的类方法与类函数:

procedure TForm1.Button1Click(Sender: TObject);
var
  MyType: TMyType;
  str: string;
begin
  MyType.Ptest;
  str := MyType.F1;
  MyType.Destroy;
end;

  2.类的静态方法

{现在的 Delphi 不仅仅有类方法, 同时有: 
  类变量: class var
  类常量: class const
  类类型: class type
  类属性: class property

  静态类方法就是给类属性来调用的, 它可以存在于私有区(private), 
  譬如下面的 SetName 就是一个静态类方法:
}
TMyClass = class(TObject)
  private
    class var FName: string;
   class procedure SetName(const Value: string); static; {静态类方法又多了一个 static 指示字}
  published
   class property Name: string read FName write SetName;
end;

 静态类方法相比类方法

  1.静态方法多了一个关键字,就是在定义的末尾加上了Static.

  2.静态方法没有隐藏的self参数,可以被当作callBack函数传给Windows API函数.

静态方法的调用约定:

  只要记住Windows API参数调用是从右往左,D的调用方式跟他是想反的.

  凡是在D里面调用Windows API函数的,肯定要加 static;stdcall ; 进行转化

  

  下面这一段转自https://blog.csdn.net/aaa000830/article/details/79986275 

-----------------------------------------------------------------------------------------------------------------------

注: 使用错误,或者在该加的地方没有加,可能会出现"privileged instruction"错误,或者地址访问错误。

常见的调用惯例有register, pascal, cdecl, stdcall, safecall。函数的调用管理决定了参数如何传递给子过程,并从堆栈中退出,以及寄存器在参数传递中的使用,错误和异常的处理。Delphi中默认的调用惯例是register。
  1) register和pascal:参数从左向右传递,也就是说最左边的参数最先求值并传入,最右边的参数最后求值和传入。cdecl,stdcall和safecall则按从右向左方向。
  2) 对于除cdecl之外的所有调用惯例,函数/过程在返回的时候要把堆栈中的参数退栈。对cdecl惯例,调用者在被调用的过程返回后执行参数退栈操作
  3) register调用惯例最多能用3个CPU寄存器来传递参数,而其它调用惯例只能通过堆栈来传递参数
  4) safecall调用惯例实现了异常的防火墙。在Windows上实现了跨进程的COM错误通知机制。
  5) register调用效率最高,因为它避免了堆栈的创建。Delphi中published属性必须是register。
  6) cdecl常用于调用C/C++编写的共享库中的函数;但是,如果要调用外部代码,那么一般要用stdcall和safecall
  7) 在Windows上,系统的API都是stdcall和safecall;在其它操作系统上通常用cdecl(注意:stdcall比cdecl效率要高)
  8) 在dual-interface(双接口)方法中必须用safecall惯例。
  9) pascal惯例是为了向后兼容;near/far/export用于16位Window编程中的函数调用,在32位的应用程序中不发挥作用,仅仅是为了向后兼容。

  10)即使静态方法只有一个参数,也要标明调用约定,因为不同的方式,汇编会最后给你弄不同的返回方式.
下表进行了总结:
Calling conventions Parameter order Clean-up Passes parameters in registers?
register Left-to-right Routine Yes
pascal Left-to-right Routine No
cdecl Right-to-left Caller No
stdcall Right-to-left Routine No
safecall Right-to-left Routine No

--------------------转载结束-----------------------------------------

 静态类的属性(class property):

  Ttest = class
  private
    class var
      FMyName: string;
  public
    class function GetMyName: string; static;
    class procedure SetMyName(Value: string); static;
  //下面这两种方法是等价的,二选一
    class property MyName: string read GetMyName write SetMyName;
    class property DirectName: string read FMyName write FMyName;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.Button1Click(Sender: TObject);
begin
  Ttest.SetMyName('张飞');
  ShowMessage(Ttest.FMyName);  //show 张飞
  ShowMessage(Ttest.GetMyName);  //show 张飞
end;

{ Ttest }
class function Ttest.GetMyName: string;
begin
  Result := fmyname;
end;

class procedure Ttest.SetMyName(Value: string);
begin
  fmyname := Value;
end;

这是说一下class var 与 var的区别.var是用来与class var做区分的,算了,我文笔不太好,还是用代码来说明吧

 还有一种方法,就是把calss var变量写在限制区域的最后面,这样就不会影响到其他字段了,比如

类的构建函数与解析函数.

  构建/解析函数优先级大于单元文件初始区(iniialization)与单元结束区(finalization).

  我们可以通过constructor和destructor关键字创建多个构建/解析函数,却不能使用class创建第二个类构建/类解析函数

  单例模式  (Singleton Pattern)

Singleton Pattern模式的设计意图是:保证一个类仅有一个实例,并提供一个访问他的全局访问点。单例模式在应用开发中比较常见,如 Application 或 Logger。在面试考试中出现率很高,别看书上一面带过,实际上还是有点复杂的,可以在B站上面搜索看看.

在 Delphi 的以前版本中,实现单例模式比较“另类”,自从 Delphi 后期加入一些新的语法元素后,单例模式的实现显得更为标准,和 C++、Java 中的实现方法几乎一致,最主要原因就是 Delphi 加入了类变量的支持,关键字为”class var”。
Delphi 同时支持类属性,可以让单例的访问更为友好;同时需要注意,在实现单例模式时,一定不要忘记把类本身的 Create 构建函数隐藏,否则的话,单例的实现将没有意义。
下面的代码实现了单例模式的 TLogger:

type
  TLogger = class ( TObject )
  private
    class var
      FInstance: TLogger;
    class function GetInstance: TLogger; static;
  protected
    constructor Create;
  public
     procedure Login;
    procedure Logout;
    class procedure ReleaseInstance;
    class property Instance: TLogger read GetInstance;
  end; 

Delphi Class of 类引用

以下内容转自Delphi Class of 类引用 - 走看看

Delphi Class of 类引用也就是类的类型,也可说是指向类的指针

Type
    TControlCls = Class of TControl;
    function CreateComponent(ControlCls: TControlCls): TControl;
    begin
        result:=ControlCls.Create(Form1);
        ...
    end;

前者要求传入一个类, 而后者要求传入一个对象(类的实例)

type
MyClassRef=calss of CMyClass //表示MyClassRef为指向CMyClass或其父类的指针

类的引用就像指向类的指针一样
类引用就是类的类型,可以声明一个类引用变量赋给它一个类,可以通过这个变量创建对象的实例。

类的类,当你不确定调用的类模型时候用到类的类。也可以说是类指针~

System单元的TObject有如下方法:
function ClassType: TClass;

它就是获取对象的类类型,它的返回类型TClass就是class of TObject。

因为所有类都派生自TObject,所以所有对象都可以调用ClassType。比如:

procedure TForm1.Button1Click(Sender: TObject);
var
S: TStringList;
C: TClass;
begin
S := TStringList.Create;
C := S.ClassType;
ShowMessage(C.ClassName);//对话框会显示出来TStringList,相当于TStringList.ClassName
S.Free
end;

而C.ClassName调用的是TObject的类方法ClassName,

原型:class function ClassName: ShortString;

就是说不需要用实例化的对象去调用,直接用类去调用就行了;不过用对象调用也是可以的,因为对象空间也保存了类的VMT地址。

用类调用形如:TStringList.ClassName
用对象调用形如:S.ClassName

一个有意思的语法 :

class help for  类助手(我不知道该叫什么,书上译为类别助手), 对现有的类进行扩展,如果把类理解为一个容器,那么这条语法的作用是对现有的类的容器的成员进行扩展或者修改。

  TTest = class    //主类,正常定义
  private
    Fnumber: Integer;
    FText: string;
  public
    FText2: string;
    procedure increase;
    constructor Create;
  end;

  TTestHelp = class helper for TTest    //类助手,关键字为class Helper for后面跟主类名
  private
    procedure Show;  //可以扩充主类的属性和方法
  public
    procedure increase;   //将会覆盖掉原有的属性
    function test(): string;  //可以扩充主类的属性和方法
    constructor Create(); //将会覆盖掉原有的方法
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

{ ttest }

constructor TTest.Create();  
begin
  inherited;
  Fnumber := 100;
  FText := '123';
  FText2 := 'TTest.Create';
  Form1.Memo1.Lines.add(FText2);
end;

procedure TTest.increase;
begin
  Inc(Fnumber);
  Form1.Memo1.Lines.add(Fnumber.ToString);
end;

{ TTestHelp }

constructor TTestHelp.Create();
begin
  inherited;
  Fnumber := 100;
  FText := 'abc';
  FText2 := 'TTestHelp.Create';
  Form1.Memo1.Lines.add(FText2);
end;

procedure TTestHelp.increase;
begin
  Dec(Fnumber);
  Form1.Memo1.Lines.add(' TTestHelp.increase:' + Fnumber.ToString);
end;

procedure TTestHelp.Show;
begin
  Form1.Memo1.Lines.add('TTestHelp.Show');
end;

function TTestHelp.test(): string;
begin
  Result := 'TTestHelp.test';
  Form1.Memo1.Lines.add(Result);
end;

//调用
procedure TForm1.Button1Click(Sender: TObject);
var
  test: TTest;
begin
  test := TTest.Create;  
  test.show;
  test.test();
end;

输出如下:

概括一下类助手的特点:

  1.主类必须是已存在的类,定义时也要注意先后顺序,先定义主类,再定义类助手,否则报错,这一点应该不难理解吧.

  2.类助手可以扩充主类的方法和属性.扩充的方法和属性可以是类方法,类变量与类属性,甚至还可以是虚拟方法!

  3.字段只能定义在主类中,类助手中不允许定义字段,否则会提报错: E2599 字段定义不允许在帮助器类型中

  4.主类和类助手是一体的,相当于把类助手的所有方法和属性都并到了主类里,同名即覆盖.

  包括构建函数和解析函数.也包括方法和属性的访问权限.比如主类里面是private,类助手是可以把它设置成public覆盖过去的.

  打个比方法,就像我们平时,把文件从A位置复制到B位置,然后中途弹出一个提示框,说存在同名文件,问你要不要覆盖,然后你选择了覆盖.

  5.引用外部资源的类助手文件,必须在USES区域中加入.

  6.编译器只承认最后一个类助手.注意!这里不是覆盖啦!是忽视最后一个类助手之前的所有类助手!!只取最后一个类助手进行覆盖!!然后你是不是想套娃?比如给类助手再配个助手?  

 TTestHelp2 = class helper for TTestHelp    //ERROR : E2021 需要class类型

  天真了吧!解决的方案之一是 真·套 "娃" ,这二是取别名(下面会讲)...重新定义一个类,让它继承原来的主类TTest,然后再给TTest2定义一个类助手,书中建议不要这么用,因为这会让类结构复杂化,源码难于解读.

   TTest2=class (TTest )
     //something
   end;
  TTestHelp2 = class helper for TTest2
  public
    procedure aaa;
  end;

  7.类助手的意义在于弥补主类功能上的不足,而不是覆盖.覆盖会使人解读混乱.希望读者能明白这一点.用下面这个例子帮助大家理解:

我们通常取ListBox当前选中的值,都是

ListBox1.Items [ListBox1.ItemIndex]

现在我觉得它使用起来太麻烦了,我需要按照我自己的方法来实现:

type
  TListboxHelper = class helper for TListBox  //扩展
    function ItemIndexValue: string;
  end;
function TListboxHelper.ItemIndexValue: string;
begin
  Result := '';
  if ItemIndex >= 0 then
    Result := Items [ItemIndex];
end;
//调用
str:= ListBox1.ItemIndexValue;

record helper for 与 class helper for相似,它是针对记录类型的.暂且叫它记录助手吧.

SysUils里的一些记录助手:

TGUIDHelper = record helper for TGUID
TStringHelper = record helper for string
TSingleHelper = record helper for Single
TDoubleHelper = record helper for Double
TExtendedHelper = record helper for Extended
TByteHelper = record helper for Byte
TShortIntHelper = record helper for ShortInt
TWordHelper = record helper for Word
TSmallIntHelper = record helper for SmallInt
TCardinalHelper = record helper for Cardinal
TIntegerHelper = record helper for Integer
TUInt64Helper = record helper for UInt64
TInt64Helper = record helper for Int64
TNativeUIntHelper = record helper for NativeUInt
TNativeIntHelper = record helper for NativeInt
TBooleanHelper = record helper for Boolean
TByteBoolHelper = record helper for ByteBool
TWordBoolHelper = record helper for WordBool
TLongBoolHelper = record helper for LongBool
TCurrencyHelper = record helper for Currency // added in Delphi 11

类型助手:

  我们上面说过,编译器对于类助手与记录助手,只会识别最后一个助手,除了真套娃,还有其他方法吗?书上说:定义一个类型别名.类型别名会被编译器当成一个全新的类型,所以它可以拥有它自己的类助手.打个比方:银行规定一个人只能开一个账户(我们抛开银行开户的逻辑不谈),张三开了一个账户之后 ,又取了个小名叫张飞又开了一个账户,这样张三就有了两个账户了(助手).当张三想用第一个账户时,他就是张三,当他想用第二的账户时,他只需要转换一下身份,变成张飞就可以了.

type
MyInt = type Integer;  //要取别名,因为系统中已存在 TIntegerHelper = record helper for Integer

TMyIntHelper = record helper for MyInt
    function AsString: string;
end;

function MyIntHelper.AsString: string;
begin
  Result := IntToStr (self);
end;

//调用
procedure TForm1.Button1Click(Sender: TObject);
var
MI: MyInt;
begin
MI := 10;

Show (MI.AsString);

// Show (MI.toString); // this doesn't work 
Show (Integer(MI).ToString)
end;

delphi D11编程语言手册 学习笔记(P344-392) 接口/类操作 - 一曲轻扬 - 博客园 (cnblogs.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/625970.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java-API简析_java.util.ArrayList类(基于 Latest JDK)(浅析源码)

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://blog.csdn.net/m0_69908381/article/details/131119289 出自【进步*于辰的博客】 其实我的【Java-API】专栏内的博文对大家来说意义是不大的。…

2023最新ChatGPT商业版4.8.8支持GPT4.0+AI绘画midjourney+实时语音识别输入+后台一键版本更新!

2023最新ChatGPT商业版4.8.8支持GPT4.0AI绘画midjourney永久更新! 1.网站系统源码介绍: 程序已支持ChatGPT4.0、Midjourney绘画、GPT3.5 API绘画、语音识别输入、用户会员套餐用户每日签到功能后台管理一键更新版本。支持手机电脑不同布局页面自适应。…

【MySQL 数据库】6、一篇文章学习【索引知识】,提高大数据量的查询效率【文末送书】

目录 一、索引概述二、索引结构(1) 不同类型的索引结构(2) 二叉树和红黑树(3) B 树(4) B树(5) Hash(6) 为什么InnoDB存储引擎选择使用Btree索引结构相对于二叉树,层级更少,搜索效率高; 三、索引的分类(1) 聚集索引和二级索引(2) 思考题 四、索…

STM32之FreeRTOS

目录 FreeRTOS 介绍 什么是 FreeRTOS ? 为什么选择 FreeRTOS ? FreeRTOS 资料与源码下载 祼机开发与 FreeRTOS 祼机开发: FreeRTOS: FreeRTOS 实现多任务的原理 二、移植 FreeRTOS 手动移植 使用CubeMX快速移植 快速移植流程…

基于Hadoop与Electron的京东商品评论词云统计系统

一、综合设计目的与要求 合肥工业大学软件工程专业《云计算、大数据技术与应用》课程综合设计报告。 爬取京东或淘宝某一商品的评论1000条,统计词频(使用MapReduce或HBase或Hive),并以词云的方式可视化呈现,最后设计为…

python-面向对象:三大特性高级特性

文章目录 前言一、面向对象三大特性:封装、继承、多态1.对象和类2.封装3.继承(1)重写父类方法(2)多继承(3)私有属性与私有方法 4.多态 二、三大特性的应用1.链表的封装2.栈的封装3.队列的封装4.二叉树的封装 三、高级特性1.类属性与实例属性2.类方法与静态方法3.pro…

三相PWM整流器滞环电流控制MATLAB仿真模型

三相PWM整流器滞环电流控制MATLAB仿真模型资源-CSDN文库https://download.csdn.net/download/weixin_56691527/87882090 模型简介: 该模型采用滞环电流控制方法来控制PWM整流器,在matlab/simulink中实现。电流内环采用三个滞环比较器,电压外…

C#,码海拾贝(38)——求解“线性方程组”的“高斯-赛德尔迭代法”之C#源代码

using System; namespace Zhou.CSharp.Algorithm { /// <summary> /// 求解线性方程组的类 LEquations /// 原作 周长发 /// 改编 深度混淆 /// </summary> public static partial class LEquations { /// <summary> /…

Nacos架构与原理 - 自研 Distro 协议 (AP分布式协议)

文章目录 背景设计思想Distro 协议工作原理数据初始化数据校验写操作读操作 小结 背景 Distro 协议是 Nacos 社区自研的⼀种 AP 分布式协议&#xff0c;是面向临时实例设计的⼀种分布式协议&#xff0c;其保证了在某些 Nacos 节点宕机后&#xff0c;整个临时实例处理系统依旧可…

VMware Workstation 17 安装教程

哈喽&#xff0c;大家好。今天一起学习的是VMware Workstation 17的安装&#xff0c;vm虚拟机是小编非常喜欢的生产力软件&#xff0c;小编之前发布的测试教程钧在vm上进行的实验。 VMware Workstation是一款功能强大的桌面虚拟计算机软件&#xff0c;它能够让用户在宿主机操作…

mac 电脑CPU温度怎么看?怎么可以监控Mac CPU温度,为什么我的 MacBook Air 这么热?

众所周知&#xff0c;电脑温度太高会直接影响到系统运行速度&#xff0c;对硬盘伤害也是很大的。特别是夏天&#xff0c;Mac 笔记本有时候运行起来会比较烫。关于 Mac 笔记本的散热&#xff0c;见仁见智。但是我们也比较好奇 CPU、电池的温度。怎么查看Mac CPU温度呢&#xff1…

【Linux工具】编译器、调式器、项目自动化构建工具以及git的使用(1编译器)

【Linux工具】编译器、调式器、项目自动化构建工具以及git的使用&#xff08;1&#xff09; 目录 【Linux工具】编译器、调式器、项目自动化构建工具以及git的使用&#xff08;1&#xff09;Linux编译器-gcc/g使用gcc的编译过程预处理&#xff08;进行宏替换&#xff09;编译&a…

2023/6/8总结

MySQL必知必会 commit 和 rollback 的差异是commit会提交&#xff0c;而rollback不会&#xff0c;就好像是撤回。 使用保留点&#xff1a; 简单的rollback和commit语句就可以写入或者撤销整个事务处理&#xff0c;但是&#xff0c;只是对简单的事务处理才能这样做&#xff0…

chatgpt赋能python:Pythontypedef:介绍及使用方法

Python typedef: 介绍及使用方法 Python作为一种高级编程语言&#xff0c;支持定义类型别名&#xff0c;可以给现有类型起一个新的名字&#xff0c;从而提高代码的可读性和可维护性。在本文中&#xff0c;我们将介绍Python中的typedef概念及其使用方法。 什么是Python typede…

BiFormer实战:使用BiFormer实现图像分类任务(二)

文章目录 训练部分导入项目使用的库设置随机因子设置全局参数图像预处理与增强读取数据设置Loss设置模型设置优化器和学习率调整算法设置混合精度&#xff0c;DP多卡&#xff0c;EMA定义训练和验证函数训练函数验证函数调用训练和验证方法 运行以及结果查看测试热力图可视化展示…

【新申请】新SRRC容易Fail点分享 2.4G和5.8G的干扰规避判定方法

新SRRC容易Fail点分享 一、2.4G 产品针对于高信道边带变严格了&#xff0c;在之前带外杂散limit为-80dBm/Hz的基础上增加了一段2483.5MHz-2500MHz限值为-40dBm/MHz, 计算可得-100dBm/Hz, 限值严格了20dB。 二、2.4G和5.8G低频发射杂散变严&#xff0c;新规2.4G和5.8G限值变为…

三相PWM整流器有限集模型预测电流控制MATLAB仿真模型

三相PWM整流器有限集模型预测电流控制MATLAB仿真模型-Matlab文档类资源-CSDN文库https://download.csdn.net/download/weixin_56691527/87881809模型简介&#xff1a; 整流器交流侧为三相对称电压&#xff0c;220V/50Hz&#xff0c;直流侧为760V&#xff0c;且电压可调。其中模…

Java 八股文-基础篇

Java 基础 一、Java 概述 1.什么是 Java&#xff1f; Java 是一门面向对象的编程语言&#xff0c;不仅吸收了 C语言的各种优点&#xff0c;还摒弃了 C里难以理解的多继承、指针等概念&#xff0c;因此 Java 语言具有功能强大和简单易用两个特征。Java 语言作为静态面向对象编…

树莓派安装 VScode 与卸载 VScode

0. 实验准备 一个带有系统的树莓派&#xff08;有屏幕更好&#xff09; 一台联网的电脑&#xff0c;且可以使用 VNC 登录树莓派&#xff08;与屏幕二选一&#xff09; 一个可以与树莓派交互文件的软件、如&#xff1a;MobaXterm&#xff08;推荐&#xff09;、WinSCP 1. 获取…

ch7_1指令系统

计算机硬件与软件之间的接口&#xff0c; 指令系统。 1.机器指令 1.1 指令的格式 指令的格式是什么&#xff1f; 操作码&#xff0c;地址码&#xff0c;寻址方式&#xff1b; 指令的字长&#xff0c;可以分为固定字长&#xff0c;可变字长&#xff1b; 操作码的长度可以是…