zoukankan      html  css  js  c++  java
  • Delphi2007新功能 -- 有限的栈对象

    今天使用Delphi2007,一个误输入,无意中发现Delphi2007的record类型居然能够和TObject一样定义方法和属性,而且不需要调用类似TObject.Create方法就能生成一个record对象。这是否意味着Delphi2007也能和C++一样,不仅能使用堆对象(new),也能使用栈对象(静态对象)呢?

            通过实验,答案是肯定的!这使我感到很兴奋,因为Delphi从诞生之日起,就限定了只能继承于TObject的堆对象,必须通过Create和Free来建立和销毁一个对象,而不能使用栈对象。而栈对象的好处就在于它能和其它类型一样很方便的定义,它能自动调用构造函数和析构函数,在作用域范围内(如函数内的局部对象)不必当心对象的建立和销毁问题。要是Delphi具有了该功能,无意是Delphi编程者们的福音。

            不过,进一步的测试却比较失望,请看下面的代码:

    type
      TMyRecord = record
      private
        x: Integer;
        y: Integer;
      public
        // error: Parameterless constructors not allowed record types
    //    constructor Create;
        // error: A record cannot introduce a destructor
    //    destructor TMyRecird;
        constructor Create(a: Integer);
        constructor TMyRecord; overload;
        constructor TMyRecord(a, b: Integer); overload;
    //    constructor TMyRecord(a, b: Integer);
        procedure Display;
        property xi: Integer read x;
      end;
    
    { TMyRecord }
    
    constructor TMyRecord.Create(a: Integer);
    begin
      x := a;
      y := 0;
    end;
    
    procedure TMyRecord.Display;
    begin
      ShowMessage(Format('x = %d, y = %d', [x, y]));
    end;
    
    constructor TMyRecord.TMyRecord;
    begin
      x := 0;
      y := 0;
    end;
    
    constructor TMyRecord.TMyRecord(a, b: Integer);
    begin
      x := a;
      y := b;
    end;
    
    procedure TForm1.Button1Click(Sender: TObject);
    var
      rec: TMyRecord;
    begin
      rec.TMyRecord(100, 200);
      rec.Display;
    end;
            可以看出,确实可以和C++一样定义一个局部record对象(不需要动态建立),而且可以使用该对象的方法和属性,请看TForm1.Button1Click方法的汇编码(CPU调试窗口粘贴下来的):
    Unit1.pas.72: begin
    0045ACDC 83C4F8           add esp,-$08
    Unit1.pas.73: rec.TMyRecord(100, 200);
    0045ACDF 8BC4             mov eax,esp
    0045ACE1 B9C8000000       mov ecx,$000000c8
    0045ACE6 BA64000000       mov edx,$00000064
    0045ACEB E8E4FFFFFF       call TMyRecord.TMyRecord
    Unit1.pas.74: rec.Display;
    0045ACF0 8BC4             mov eax,esp
    0045ACF2 E855FFFFFF       call TMyRecord.Display
    Unit1.pas.75: end;
    0045ACF7 59               pop ecx
    0045ACF8 5A               pop edx
    0045ACF9 C3               ret

    从汇编码中看,定义的TMyRecord变量rec确实是标准C++栈对象一样的,在调用其方法的时候也隐含传递了Self指针(mov  eax,esp),也就是说,rec在这里已经区别于普通的record变量,而是一个实实在在的“对象”变量!

            但是遗憾的是,它不能定义destructor方法,虽然可以定义constructor方法(不能定义为Tobject缺省的Create方法,带参数可以),但却不能在定义的时候自动调用构造方法(我原本以为和C++一样,定义一个和TMyRecord同名的constructor方法,可以自动调用的),如此一来,静态栈对象的主要优点就丧失了,如果手工调用构造方法,和定义一个普通record变量,然后调用静态过程没什么两样了。其实,Delphi已经把record改道这一步了,稍稍进一步就可以完完全全的使用栈对象,从技术上讲一点难度都没有,编译器只需要在栈中分配对象变量内存和销毁对象变量内存的同时,隐含调用一下构造方法和析构方法就可以了的。

            虽然很遗憾,但是还是为Delphi的这点进步而高兴,至少我们可以利用这一功能封装record,而且使用这有限的栈对象可以不用当心对象本身的销毁问题(record对象中动态分配的内存还是得手工释放或者调用它的某个方法释放)。

            后记:以前在DOS下,以C/C++为主,Pascal只用过Turbo Pascal4.0,对以后Pascal的版本并不十分了解,今天看了网友housisong的留言,才知道我所谓的新功能--有限栈对象,Delphi中早就有了,如Delphi7.0就可以使用Object关键字定义栈对象,而且比2007的record类型还完善一些,record不能继承,而object对象能够继承,看来我真是孤陋寡闻了!!!

  • 相关阅读:
    [转]protobuf的编译安装
    [转]OpenMP中几个容易混淆的函数(线程数量/线程ID/线程最大数)以及并行区域线程数量的确定
    C++类中静态数据成员MAP如何初始化
    [转]gcc -ffunction-sections -fdata-sections -Wl,–gc-sections 参数详解
    的机器学习开源工具分享
    机器学习常见的采样方法
    图像配准与深度学习方法
    卷积网络中的几何学你了解多少?
    云计算、虚拟化和容器
    在数据科学领域,你掌握这个24个python库就够了!
  • 原文地址:https://www.cnblogs.com/del88/p/5407140.html
Copyright © 2011-2022 走看看