(delphi11最新学习资料) Object Pascal 学习笔记---第11章第1节 ( 接口的实现)

2024-05-06 07:44

本文主要是介绍(delphi11最新学习资料) Object Pascal 学习笔记---第11章第1节 ( 接口的实现),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

11.1.2 接口的实现

​ 任何类都可以实现一个或多个接口,方法是在继承的基类后面列出这些接口,并为每个接口的每个方法提供一个实现:

typeTAirplane = class(..., ICanFly)function Fly: string;end;function TAirplane.Fly: string;
begin// 实际代码
end;

​ 当一个类实现一个接口时,这个类必须实现接口中所有的方法,并且方法签名要完全相同。如在本例中,TAirplane 类就必须将 Fly 方法实现为返回字符串的函数。由于接口也继承自一个基接口(IInterface),因此实现接口的类必须全部实现接口及基接口中的所有方法。

​ 这就是为什么一个类继承一个基类来实现接口的方式非常常见,因为这个基类已经实现了 IInterface 基接口方法。Object Pascal 运行库已经提供了这样一些基类来实现基本行为。这些类中最简单的是 TInterfacedObject 类,因此上面的代码可以变成:

typeTAirplane = class(TInterfacedObject, ICanFly)function Fly: string;end;

注解 实现接口时,可以使用静态方法或虚方法。如果你打算覆盖继承类中的方法,使用虚拟方法是合理的。不过也有另一种方法,那就是指定基类也继承自相同的接口,然后覆盖该接口的方法。我倾向于在需要时将实现接口方法的方法声明为虚方法。

​ 现在我们已经定义了一个接口及其实现的类,我们可以创建这个类的对象。我们可以将其视为普通类,如下所示:

varAirplane1: TAirplane;
beginAirplane1 := TAirplane.Create;tryAirplane1.Fly;finallyAirplane1.Free;end;
end;

​ 在这种情况下,我们可以忽略类实现了接口这一事实。不同之处在于,现在我们还可以声明一个接口类型的变量。使用接口类型变量会自动启用引用内存模型,因此我们可以跳过 try-finally 块:

varFlyer1: ICanFly;
beginFlyer1 := TAirplane.Create;Flyer1.Fly;
end;

​ 对于这个看似简单的代码片段(也是 Intf101 示例的一部分)的第一行,有几个相关的注意事项。首先,只要将对象赋值给接口变量,运行时就会使用特殊版本的 as 操作符自动检查对象是否实现了该接口。您可以通过编写下面的同一行代码显式地进行这一操作:

Flyer1 := TAirplane.Create as ICanFly;

​ 其次,无论我们使用直接赋值还是 as 语句,运行时都会做一件额外的事情:调用对象的 _AddRef 方法,增加对象的引用计数。这是通过调用对象从基类 TInterfacedObject 继承的方法完成的。

​ 与此同时,一旦 Flyer1 变量离开作用域(即执行结束语句时),Delphi 运行时就会调用 _Release 方法,该方法会减少引用计数,检查引用计数是否为零,如果为零就销毁对象。这就是为什么在上面的代码中,我们不需要手动释放在代码中创建的对象,也不需要编写 try-finally 块。

注解 虽然上面的源代码没有 try-finally 代码块,但编译器会自动在方法中添加一个隐式 try-finally 代码块,并隐式调用 _Release。这种情况在 Object Pascal 中很多:基本上每当一个方法有一个或多个托管类型(如字符串、接口或动态数组)时,编译器都会自动添加一个隐式的 try-finally 代码块。

​ 正如我们在上面的代码中所看到的,被接口变量引用的 Object Pascal 对象是被引用计数的(除非接口类型变量被标记为弱变量或不安全变量,这将在后面解释)。我们还看到,当不再有接口变量引用这些对象时,它们会被自动回收。
​ 值得注意的是,虽然编译器会使用一些魔法(隐藏的 _AddRef 和 _Release 调用),但实际的引用计数机制取决于开发者或运行时库提供的具体实现。在上一个示例中,由于 TInterfacedObject 类方法中的代码(此处列出的是略微简化的版本),引用计数确实在起作用:

function TInterfacedObject._AddRef: Integer;
beginResult := AtomicIncrement(FRefCount);
end;function TInterfacedObject._Release: Integer;
beginResult := AtomicDecrement(FRefCount);if Result = 0 thenbeginDestroy;end;
end;

​ 现在来看看RTL中另外一个实现 IInterface 接口的基类 TNoRefCountObject,该类基本上禁用了实际的引用计数机制:

function TNoRefCountObject._AddRef: Integer;
beginResult := -1;
end;function TNoRefCountObject._Release: Integer;
beginResult := -1;
end;

​ Delphi 11 中引入了TNoRefCountObject 新类,该类的对象可以忽略引用计数机制。TNoRefCountObject 类用来取代名称奇怪的 TSingletonImplementation 类(定义在 Generics.Defaults 单元中)。旧的 TSingletonImplementation 类仍然作为较新的TNoRefCountObject 类的别名存在。这两个类的代码基本相同。改变的主要原因是,原来的类名不副实,因为它与单例模式毫无关系。我们将在下一章中看到单例模式的示例。

​ 虽然 TNoRefCountObject 并不常用,但还有一个类TComponent也实现了接口并禁用了引用计数机制,这只是因为它需要有自己的内存管理模式。如果你想拥有一个实现接口的自定义组件,就不必担心引用计数和内存管理问题。在本章最后的 "使用接口实现设计模式 "一节中,我们将看到一个自定义组件实现接口的例子。

这篇关于(delphi11最新学习资料) Object Pascal 学习笔记---第11章第1节 ( 接口的实现)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/963798

相关文章

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java easyExcel实现导入多sheet的Excel

《JavaeasyExcel实现导入多sheet的Excel》这篇文章主要为大家详细介绍了如何使用JavaeasyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录1.官网2.Excel样式3.代码1.官网easyExcel官网2.Excel样式3.代码

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

浏览器插件cursor实现自动注册、续杯的详细过程

《浏览器插件cursor实现自动注册、续杯的详细过程》Cursor简易注册助手脚本通过自动化邮箱填写和验证码获取流程,大大简化了Cursor的注册过程,它不仅提高了注册效率,还通过友好的用户界面和详细... 目录前言功能概述使用方法安装脚本使用流程邮箱输入页面验证码页面实战演示技术实现核心功能实现1. 随机

Golang如何对cron进行二次封装实现指定时间执行定时任务

《Golang如何对cron进行二次封装实现指定时间执行定时任务》:本文主要介绍Golang如何对cron进行二次封装实现指定时间执行定时任务问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录背景cron库下载代码示例【1】结构体定义【2】定时任务开启【3】使用示例【4】控制台输出总结背景