14

关于德尔福记录的一些问题:

  1. 由于记录几乎就像类,为什么不只使用类而不使用记录呢?
  2. 理论上,当一条记录被一个变量声明时,就会为它分配内存;但是,之后如何释放内存?
  3. 我可以理解指向记录的指针到列表对象的效用,但是使用泛型容器(TList<T>),还需要使用指针吗?如果没有,如何将每条记录删除/释放到通用容器中?如果我想将特定记录删除到通用容器中,该怎么做?
4

5 回答 5

19

记录和类之间有很多差异;并且没有“记录指针” <> “类”。每个都有自己的优点和缺点;关于软件开发的重要事情之一是了解这些,以便您可以更轻松地选择最适合给定情况的方法。

  1. 这个问题是基于一个错误的前提。记录几乎不像类,就像整数不像双精度数一样。
    • 类必须始终动态实例化,虽然这是可能的,但不是对记录的要求。
    • 类的实例(我们称为对象)总是通过引用传递,这意味着多个代码部分将共享并作用于同一个实例。记住这一点很重要,因为您可能会无意中修改对象作为副作用;尽管故意这样做是一个强大的功能。另一方面,记录是按值传递的;您需要明确指出您是否通过引用传递它们。
    • 类不会“像记录一样容易复制”。当我说复制时,我的意思是复制源的单独实例。(根据上面的价值/参考评论,这应该是显而易见的)。
    • 记录往往与键入的文件一起工作得很好(因为它们很容易复制)。
    • 记录可以将字段与其他字段重叠(案例 x of/unions)
    • 这些是对记录的某些情境益处的评论;相反,我不会详细说明的课程也有情境优势。
  2. 也许理解这一点的最简单方法是对它有点迂腐。让我们澄清一下;内存不是在“声明时”真正分配的,而是在变量在范围内时分配的,在超出范围时释放。因此,对于局部变量,它在例程开始之前分配,并在结束之后释放。对于类字段,它在对象创建时分配,在对象销毁时释放。
  3. 再次,有利有弊...
    • 与仅复制引用相比,复制整个记录(与泛型一样)可能更慢并且需要更多内存。
    • 通过引用传递记录(使用指针)是一种强大的技术,您可以轻松地让其他东西修改您的记录副本。如果没有这个,您必须按值传递您的记录(即复制它),结果接收更改的记录,再次将其复制到您自己的结构中。
  4. 指向记录的指针是否像类?一点都不。只有两个区别:
    • 类支持多态继承。
    • 类可以实现接口。
于 2009-12-09T23:47:39.907 回答
11

对于 1 和 2:记录是值类型,而类是引用类型。它们是在堆栈上分配的,或者直接在包含它们的任何较大变量的内存空间中分配,而不是通过指针分配,并在它们超出范围时由编译器自动清理。

至于你的第三个问题, aTList<TMyRecord>内部声明了一个array of TMyRecordfor 存储空间。当列表被销毁时,其中的所有记录都将被清理。如果要删除特定的,可以使用Delete按索引删除的Remove方法,或者查找删除的方法。但请注意,由于它是一种值类型,因此您所做的一切都将复制记录,而不是复制对它的引用。

于 2009-12-09T21:16:48.390 回答
9

记录的主要好处之一是,当您拥有大量“记录数组”时。这是通过在一个连续的 RAM 空间中为所有记录分配空间而在内存中创建的,这非常快。如果您改用“TClass 数组”,则数组中的每个对象都必须自己分配,这很慢。

为了提高字符串和对象的速度,已经做了很多工作来提高分配内存的速度,但永远不会像用 1 个内存分配替换 100,000 个内存分配那样快。

但是,如果您使用记录数组,请不要在局部变量中复制记录。这可能很容易扼杀速度优势。

于 2009-12-10T06:45:55.480 回答
3

类和记录之间还有一些其他差异。类可以使用多态性,并公开接口。记录不能实现析构函数(尽管从 Delphi 2006 开始,它们现在可以实现构造函数和方法)。

记录在将内存分段成更符合逻辑的结构时非常有用,因为记录中的第一个数据项与指向记录本身的指针的地址点相同。这不是类的情况。

于 2009-12-09T23:21:41.073 回答
3

1)为了允许继承和多态,类有一些开销。记录不允许使用它们,并且在某些情况下使用起来可能更快更简单。与始终在堆中分配并通过引用管理的类不同,记录也可以在堆栈上分配、直接访问并相互分配,而无需调用“分配”方法。记录对于访问具有给定结构的内存块也很有用,因为它们的内存布局正是您定义它的方式。类实例内存布局由编译器控制,并具有使对象工作的附加数据(即指向虚拟方法表的指针)。

2) 除非您使用 New() 或 GetMem() 动态分配记录,否则记录的内存由编译器作为序数、浮点数或静态数组管理:全局变量内存在启动时分配并在程序终止时释放,局部变量在在堆栈上分配进入函数/过程/方法并释放退出。在堆栈中分配/释放内存更快,因为它不需要调用内存管理器,只需很少的汇编指令即可更改堆栈寄存器。但请注意,在堆栈上分配大型结构可能会导致堆栈溢出,因为最大堆栈大小是固定的并且不是很大(请参阅链接器选项)。如果记录是类的字段,则在创建类时分配它们,并在释放类时释放它们。

3)泛型的优点之一是消除了对低级指针管理的需要——但要注意内部工作原理。

于 2009-12-10T08:52:15.660 回答