Delphi中的数据类型

之前一直使用的是Delphi 7,现在准备转向Delphi XE7,据说数据类型上有一些微小的调整,便想亲自验证一下,也顺便加深一下自己对Delphi数据类型的认识,毕竟之前仅仅是在用而从未考虑过类型在内存中的形态。

验证方法很简单,就是在控制台用Sizeof()把相应数据类型的字节数显示出来。源码如下:

program TypeSize;

{$APPTYPE CONSOLE}

uses
  SysUtils;

type
  Emnu = (one, two, three, four);

  Range1 = $0..$FF;

  Range2 = $0..$FFFF;

  Range3 = $0..$FFFFFFFF;

  arr1 = array of Byte;

  arr2 = array[0..9] of Byte;

  arr3 = array[0..9] of Integer;

  rcd1 = record
  end;

  rcd2 = record
    b: Byte;
  end;

  rcd3 = record
    i: Integer;
  end;

  rcd4 = record
    i: Integer;
    j: Integer;
  end;

  rcd5 = record
    b: Byte;
    i: Integer;
  end;

  rcd6 = record
    r: Real;
  end;

  rcd7 = record
    r: Real;
    i: Integer;
  end;

  obj1 = object
  end;

  obj2 = object
    c: Char;
  end;

  obj3 = object
    i: Integer;
    o: obj2;
  end;

begin
  Writeln('   Type     | Bytes');
  Writeln('--------------------------------');
  Writeln('AnsiChar    = ', SizeOf(AnsiChar));
  Writeln('WideChar    = ', SizeOf(WideChar));
  Writeln('Char        = ', SizeOf(Char));
  Writeln('Byte        = ', SizeOf(Byte));
  Writeln('ShortInt    = ', SizeOf(ShortInt));
  Writeln('SmallInt    = ', SizeOf(SmallInt));
  Writeln('LongInt     = ', SizeOf(LongInt));
  Writeln('Int64       = ', SizeOf(Int64));
  Writeln('Byte        = ', SizeOf(Byte));
  Writeln('Word        = ', SizeOf(Word));
  Writeln('LongWord    = ', SizeOf(LongWord));
  Writeln('Integer     = ', SizeOf(Integer));
  Writeln('Cardinal    = ', SizeOf(Cardinal));
  Writeln('Boolean     = ', SizeOf(Boolean));
  Writeln('ByteBool    = ', SizeOf(ByteBool));
  Writeln('WordBool    = ', SizeOf(WordBool));
  Writeln('LongBool    = ', SizeOf(LongBool));
  Writeln('Single      = ', SizeOf(Single));
  Writeln('Real48      = ', SizeOf(Real48));
  Writeln('Double      = ', SizeOf(Double));
  Writeln('Extended    = ', SizeOf(Extended));
  Writeln('Comp        = ', SizeOf(Comp));
  Writeln('Currency    = ', SizeOf(Currency));
  Writeln('Real        = ', SizeOf(Real));
  Writeln('ShortString = ', SizeOf(ShortString));
  Writeln('AnsiString  = ', SizeOf(AnsiString));
  Writeln('WideString  = ', SizeOf(WideString));
  Writeln('String      = ', SizeOf(string));
  Writeln('Variant     = ', SizeOf(Variant));
  Writeln('Pointer     = ', SizeOf(Pointer));
  Writeln('--------------------------------');
  Writeln('Emnu        = ', SizeOf(Emnu));
  Writeln('Range(0,FF)       = ', SizeOf(Range1));
  Writeln('Range(0,FFFF)     = ', SizeOf(Range2));
  Writeln('Range(0,FFFFFFFF) = ', SizeOf(Range3));
  Writeln('Array()           = ', SizeOf(arr1));
  Writeln('Array(10)Byte     = ', SizeOf(arr2));
  Writeln('Array(10)Int      = ', SizeOf(arr3));
  Writeln('Record()                   = ', SizeOf(rcd1));
  Writeln('Record(1 byte)             = ', SizeOf(rcd2));
  Writeln('Record(1 int)              = ', SizeOf(rcd3));
  Writeln('Record(2 int)              = ', SizeOf(rcd4));
  Writeln('Record(1 byte, 1 int)      = ', SizeOf(rcd5));
  Writeln('Record(1 real)             = ', SizeOf(rcd6));
  Writeln('Record(1 real, 1 int)      = ', SizeOf(rcd7));
  Writeln('Object()                   = ', SizeOf(obj1));
  Writeln('Object(1 char)             = ', SizeOf(obj2));
  Writeln('Object(1 int, 1 obj(char)) = ', SizeOf(obj3));
  Readln;
end.

分别用Delphi 7和Delphi XE7编译,运行结果如图:

从运行结果可以看出:

  1. Delphi 7 和 Delphi XE7 数据类型的唯一区别就是默认的字符集发生的变化,Delphi 7 中 Char = AnsiChar,Delphi XE7 中Char = WideChar,即 Delphi 7 默认使用 ANSI 编码,Delphi XE7 默认使用 Unicode 编码。相应的,Delphi XE7 中 String = WideString,PChar = PWideChar。
  2. AnsiString、WideString、String 和 Pointer 一样占用 4B 的空间,说明,字符串类型的本质其实就是指针。由于 Delphi 对字符封闭得很好,使用起来感觉和整型、浮点型等没有什么差别,但毕竟还是不一样的,这是需要注意的(有关情况我会在另一篇中详细说明)。
  3. 枚举类型4个元素占1B,其它数量时也是1B,说明枚举类型实际存储的是元素的索引,或者说枚举类型的元素实际是对应索引值的一个别名。至于超过256个元素的情况未进行测试,即枚举类型元素上限未测。
  4. 子界类型与枚举类型比较类似,当元素个数少于256时用1B存储,超过256个时用2B,2B不够时用4B(没有3B的情况),以此推测,4B不够时将用8B、16B等。
  5. 静态数组占用的是连续的内存区域,其大小为数组元素大小的总合;动态数组在未指定数组大小时只有4B,说明动态数组的本质也是指针(有关细节会同字符串一同说明)。
  6. 记录类型的大小一般来说是其元素类型的大小*元素的个数,无元素时0B;如果元素类型不一致,元素类型统一按较大的类型的大小进行存储(猜想是为了方便寻址,拿空间换时间,提高运行速度)。
  7. Object类型是Delphi较早的一种类型,与记录类型很类似,不同的是其占用空间的大小为其元素大小的总合。估计Object类型就是记录类型的前身,而在记录类型出现后也依然没有消亡,想必仅仅是为了兼容古老的程序吧。

猜你喜欢

转载自my.oschina.net/u/209067/blog/1507343