(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

相关文章

MySQL 多列 IN 查询之语法、性能与实战技巧(最新整理)

《MySQL多列IN查询之语法、性能与实战技巧(最新整理)》本文详解MySQL多列IN查询,对比传统OR写法,强调其简洁高效,适合批量匹配复合键,通过联合索引、分批次优化提升性能,兼容多种数据库... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

mybatis执行insert返回id实现详解

《mybatis执行insert返回id实现详解》MyBatis插入操作默认返回受影响行数,需通过useGeneratedKeys+keyProperty或selectKey获取主键ID,确保主键为自... 目录 两种方式获取自增 ID:1. ​​useGeneratedKeys+keyProperty(推

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构