(delphi11最新学习资料) Object Pascal 学习笔记---第13章第3节 (仅销毁对象一次 )

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

13.3.2 仅销毁对象一次

​ 另一个问题是,如果调用对象的析构函数两次,将会导致错误。析构函数是释放对象内存的方法。我们可以为析构函数编写代码,通常是覆盖默认的 Destroy 析构函数,以便让对象在被销毁之前执行一些代码。

DestroyTObject类的虚析构函数。大多数需要在对象销毁时执行自定义清理代码的类都会覆盖此虚方法。您绝对不应该定义新的析构函数,因为通常对象是通过调用Free方法来销毁的,而Free方法会为您调用Destroy虚析构函数(可能是重载版本)。

​ 正如我刚提到的,Free只是TObject类的一个方法,被所有其他类继承。Free方法基本上在调用Destroy虚析构函数之前检查当前对象(Self)是否为nil

注解:您可能会想知道为什么如果对象引用为nil,您仍然可以安全地调用Free,但不能调用Destroy。原因是Free是一个位于给定内存位置的已知方法,而Destroy虚函数是在运行时查看对象的类型来确定的,如果对象不存在,这个操作就非常危险

​ 以下是Free的伪代码:

procedure TObject.Free;
beginif Self <> nil thenDestroy;
end;

​ 接下来,我们可以将注意力转向Assigned函数。当我们将指针传递给此函数时,它只是测试指针是否为nil。因此,以下两个语句在大多数情况下是等效的:

if Assigned(MyObj) then...
if MyObj <> nil then...

​ 请注意,这些语句仅测试指针是否不为nil;它们不检查它是否是有效的指针。如果您编写以下代码:

MyObj.Free;  //不会把MyObj设置为nil
if MyObj <> nil thenMyObj.DoSomething;

测试将评估为True,并且在调用对象方法的行中会出现错误。需要注意的是,调用Free不会将对象的引用设置为nil

​ 自动将对象设置为nil是不可能的。您可能有多个引用指向同一个对象,而Object Pascal 不会跟踪它们。同时,在方法中(例如Free方法),我们可以操作对象,但我们对对象引用一无所知——即我们用于调用该方法的变量的内存地址。

​ 换句话说,在Free方法或类的其他任何方法中,我们知道对象(Self)的内存地址,但我们不知道引用对象的变量的内存位置,比如MyObj。因此,Free方法无法影响MyObj变量。

​ 然而,当我们将对象作为按引用传递参数的方式调用一个外部函数时,该函数随后可以选择修改原始对象引用。这正是FreeAndNil过程所做的事情,FreeAndNil过程可以替代使用Free然后将引用变量设置为nil。以下是FreeAndNil的代码:

procedure FreeAndNil(const [ref] Obj: TObject); inline;
varTemp: TObject;
beginTemp := Obj;TObject(Pointer(@Obj)^) := nil;Temp.Free;
end;

​ 在过去,参数只是一个指针,缺点是您可以将原始指针、接口引用和其他不兼容的数据结构传递给FreeAndNil过程。这通常会导致内存损坏和难以发现的错误。从Delphi 10.4开始,代码已被修改如上所示,使用TObject类型的const引用参数,将参数限制为对象。

注解:许多Delphi专家会争论,FreeAndNil永远不应该使用,因为引用对象的变量的可见性应该与其生命周期相匹配。如果对象拥有另一个对象并在析构函数中释放它,就不需要将引用设置为nil,因为它是不再使用的对象的一部分。同样,具有在try-finally块中释放的局部变量也不需要将其设置为nil,因为它即将退出作用域。

​ 顺便提了一下,除了Free方法之外,TObject还有一个DisposeOf方法,它是多年以前Object Pascal语言支持ARC的产物。目前,DisposeOf方法只是调用Free

​ 总结一下关于这些内存清理操作的使用,这里是一些建议:

  • 始终调用Free来销毁对象,而不是调用Destroy析构函数。
  • 在调用Free之后使用FreeAndNil,或将对象引用设置为nil,除非引用紧接着超出作用域。

这篇关于(delphi11最新学习资料) Object Pascal 学习笔记---第13章第3节 (仅销毁对象一次 )的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

在java中如何将inputStream对象转换为File对象(不生成本地文件)

《在java中如何将inputStream对象转换为File对象(不生成本地文件)》:本文主要介绍在java中如何将inputStream对象转换为File对象(不生成本地文件),具有很好的参考价... 目录需求说明问题解决总结需求说明在后端中通过POI生成Excel文件流,将输出流(outputStre

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

查看Oracle数据库中UNDO表空间的使用情况(最新推荐)

《查看Oracle数据库中UNDO表空间的使用情况(最新推荐)》Oracle数据库中查看UNDO表空间使用情况的4种方法:DBA_TABLESPACES和DBA_DATA_FILES提供基本信息,V$... 目录1. 通过 DBjavascriptA_TABLESPACES 和 DBA_DATA_FILES

最新Spring Security实战教程之Spring Security安全框架指南

《最新SpringSecurity实战教程之SpringSecurity安全框架指南》SpringSecurity是Spring生态系统中的核心组件,提供认证、授权和防护机制,以保护应用免受各种安... 目录前言什么是Spring Security?同类框架对比Spring Security典型应用场景传统

最新Spring Security实战教程之表单登录定制到处理逻辑的深度改造(最新推荐)

《最新SpringSecurity实战教程之表单登录定制到处理逻辑的深度改造(最新推荐)》本章节介绍了如何通过SpringSecurity实现从配置自定义登录页面、表单登录处理逻辑的配置,并简单模拟... 目录前言改造准备开始登录页改造自定义用户名密码登陆成功失败跳转问题自定义登出前后端分离适配方案结语前言

OpenManus本地部署实战亲测有效完全免费(最新推荐)

《OpenManus本地部署实战亲测有效完全免费(最新推荐)》文章介绍了如何在本地部署OpenManus大语言模型,包括环境搭建、LLM编程接口配置和测试步骤,本文给大家讲解的非常详细,感兴趣的朋友一... 目录1.概况2.环境搭建2.1安装miniconda或者anaconda2.2 LLM编程接口配置2

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

Java实现将byte[]转换为File对象

《Java实现将byte[]转换为File对象》这篇文章将通过一个简单的例子为大家演示Java如何实现byte[]转换为File对象,并将其上传到外部服务器,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言1. 问题背景2. 环境准备3. 实现步骤3.1 从 URL 获取图片字节数据3.2 将字节数组

Javascript访问Promise对象返回值的操作方法

《Javascript访问Promise对象返回值的操作方法》这篇文章介绍了如何在JavaScript中使用Promise对象来处理异步操作,通过使用fetch()方法和Promise对象,我们可以从... 目录在Javascript中,什么是Promise1- then() 链式操作2- 在之后的代码中使