zoukankan      html  css  js  c++  java
  • [转载]delphi中的变体记录

    变体记录,我一直不是很明白,先转过去,仔细研究下
    原文地址:delphi中的变体记录作者:大海
    delphi中的变体记录
    //Integer类型刚好是4个字节,ShortInt类型是1个字节,但是Windows中内存是4字节分配,
      
    //所以这里其实还是4个字节,用SizeOf可以看到这个record的大小是8字节,这样虽然浪
      
    //费了空间,但是加快了速度(Windows内存分配中的边界对齐原理)
      TPerson 
    = record
        Age: Integer;
        Sex: ShortInt;
      
    end;

      TPackedPerson 
    = packed record
        Age: Integer;
        Sex: ShortInt;  
    //使用packed record,用Sizeof可以看到这个record的大小是5字节;
      
    end;

      TEmployee 
    = record
        ID: Integer;                       
    //Integer是4字节
        
    case Integer of
          
    0:(YearMoney: Integer);          //YearMoney和MonthMoney共用内存,按最大内存分配
          
    1:(MonthMoney: ShortInt);        //该record的大小是8字节
      
    end;

      TTagEmployee 
    = record
        ID: Integer;
        
    case Grade: Integer of              //这里加入了Grade变量
          
    0:(YearMoney: Integer);           //YearMoney和MonthMoney共用内存,按最大内存分配
          
    1:(MonthMoney: ShortInt);         //该record的大小是12字节(ID+Grade+YearMoney)
      
    end;
    一、DELPHI中记录的存储方式
    在DELPHI中,我们用record关键字来表明一个记录,有时候,我们还会看到用packed record来声明的记录,这二者的区别就在于存储方式的不同。
    在windows中,内存的分配一次是4个字节的,而Packed按字节进行内存的申请和分配,这样速度要慢一些,因为需要额外的时间来进行指针的定位。因此如果不用Packed的话,Delphi将按一次4个字节的方式申请内存,因此如果一个变量没有4个字节宽的话也要占4个字节!这样浪费了一些空间,但提高了效率。所谓鱼与熊掌不可兼得。
    例如一个记录,以,sizeof(wudi)应该得到8。而如果使用packed关键字,那么sizeof(wudi)则得到5。

    type wudi= record
    age : integer;
    sex : shortint;
    end;

    其中age是integer类型,正好4个字节,而sex是showint类型,占用一个字节,但基于4字节得内存分配方式,这里它也将占用4个字节。所

    二、DELPHI中的变体记录
    在DELPHI中,观察Tmessage和TTypeData的定义的定义,从关键字record,你一眼就可以看出,它是一个记录类型,但仔细观察,你又会发现在它的定义中出现了case关键字。它代表什么呢?
    它代表此记录是变体记录。让我们先去了解一下变体记录。

    2.1一个典型的变体记录定义如下:

    type recordTypeName = record
    fieldList1: type1;
    ...
    fieldListn: typen;
    case tag: ordinalType of
    constantList1: (variant1);
    ...
    constantListn: (variantn);
    end;

    其中case到结尾部分定义了多个变体字段。所有变体字段共享一段内存,大小由最大变体字段决定。

    2.2使用变体记录时要注意:
    (1)Long String、WideString、Dynamic Array、Interface的大小都是指针大小, OleVariant其实就是COM SDK中的VARIANT结构,大小是16字节。但在Object Pascal中它们都需要自动终结化,如果它们出现在variant part中,编译器就无法知道它们是否应该进行终结化――因为不知道当前存储的是哪种类型。
    (2)所有变体字段共享一段内存。而共享内存的大小则由最大变体字段决定。
    (3)当tag存在时,它也是记录的一个字段。也可以没有tag。
    (4)记录的变体部分的条件域必须是有序类型
    (5)记录类型中可以含有变体部分,有点象case语句,但没有最后的end,变体部分必需在记录中其他字段的声明之后

    2.3变体记录得作用
    (1)节约空间。对于那些要根据条件而决定是否存储得类型,完全可以利用变体记录来达到节约空间得效果。例如,一个公司的员工薪水可以是月薪、年薪等方式,那么并没有必要在记录中都分配空间而又用不到。
    (2)类型的转换。例如,如果有一个64位的整数类型作为变体的第一个字段,一个32位的整数Integer类型作为另一个变体的第一个字段,那么可以向64字段赋值然后以整数Integer字段读出其前32位
    (3)妙用还有很多,根据它的原理挖掘吧

    2.4一个演示变体记录使用的例子

    ...{
    作者 :wudi_1982
    联系方式:wudi_1982@hotmail.com
    转载请著名出处
    }
    unit Unit1;

    interface

    uses
    Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
    Dialogs,typinfo, StdCtrls, ComCtrls, ExtCtrls;

    type
    WdFileType = (Files, Dirs);//类型信息

    TWdFileInfo = record//一个变体记录
    Fname: string[10];//文件名
    case typeid:WdFileType of//根据是文件夹还是文件来决定
    Files : (Fsize : integer);//文件大小
    Dirs : (Dsize : integer;//文件夹大小
    ContainFileCount : integer;//包含文件的个数
    ContainDirCount : integer);//包含文件夹的个数
    end;
    TForm1 = class(TForm)
    Memo1: TMemo;
    Panel1: TPanel;
    Button1: TButton;
    GroupBox1: TGroupBox;
    Label1: TLabel;
    procedure FormCreate(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    private
    pp : array[0..1] of TWdFileInfo;//一个变体记录数组
    procedure ShowInfo; //在memo中显示当前变体记录的信息
    public
    end;

    var
    Form1: TForm1;

    implementation

    ...{$R *.dfm}

    procedure TForm1.FormCreate(Sender: TObject);
    begin
    //初始化是为记录赋值
    with pp[0] do
    begin
    Fname := 'PP0';
    typeid := Files;
    //fsize和dsize实际用的是同一块存储空间,
    //这里两个都赋值了,存储空间中记录的自然是后者
    Fsize := 10;
    Dsize := 100;
    ContainFileCount := 1;
    ContainDirCount := 2;
    end;
    with pp[1] do
    begin
    Fname := 'PP1';
    typeid := Dirs;
    Fsize := 0;
    Dsize := 100;
    ContainFileCount := 11;
    ContainDirCount := 222;
    end;
    Memo1.Clear;
    ShowInfo;
    end;

    procedure TForm1.ShowInfo;
    var
    i : integer;
    fs : string;
    begin
    Memo1.Lines.Add('变体记录信息显示');
    for i := 0 to 1 do
    begin
    fs := Format('PP%d的大小是(%d)'+#13+'Fsize:(%d),Dsize:(%d)'+#13+'ContainFileCount:(%d),CDirCount:(%d);'
    ,[i,sizeof(pp[i]),pp[i].Fsize,pp[i].Dsize,pp[i].ContainFileCount,pp[i].ContainDirCount]);
    Memo1.Lines.Add(fs);
    Memo1.Lines.Add('------------------------');
    end;


    end;

    procedure TForm1.Button1Click(Sender: TObject);
    begin
    Memo1.Clear;
    end;

    end.

  • 相关阅读:
    IIS文件大小限制
    XAMPP 配置多端口 多站点
    C# 复制文件和文件夹
    Windows下 Python 安装包的配置
    从数据库读取数据绑定到TreeView(内含设置样式,图片)
    异步请求数据简单例子
    Jmeter使用_StringFromFile函数需要添加编码方式
    利用Fitnesse和Jmeter实现接口性能测试
    简易覆盖率信息收集框架
    如何对遗留代码进行单元测试(scrumgathering听后感)
  • 原文地址:https://www.cnblogs.com/luckForever/p/7255158.html
Copyright © 2011-2022 走看看