Delphi 的字符及字符串[3] - String 中的秘密

本文详细解析了Delphi中字符串的内存布局与管理机制,包括指针地址与实际内存地址的区别,以及字符串如何进行内存分配与引用计数。

======================================================
注:本文源代码点此下载
======================================================

//string 的指针地址及实际的内存地址

var

str: string;

pstr: pstring;

pc: pchar;

begin

{在没有给 str 赋值以前, 既然声明了, 就有了指针地址(@str):}

showmessage(inttostr(integer(@str))); {1244652; 这是在栈中的 str 的指针地址}

{但现在还没有分配真正储存字符串内存}

showmessage(inttostr(integer(str)));{0; 0 就是 null}

str := 'delphi';

{一旦赋值后...}

showmessage(inttostr(integer(@str))); {1244652; 这是在栈中的 str 的指针地址}

showmessage(inttostr(integer(str)));{4580800; 这是在堆中的 str 的实际地址}

{通过指针地址获取字符串, 其中的 pstr 是前面定义的字符串指针}

pstr := @str;

showmessage(pstr^); {delphi}

{通过实际地址获取字符串, 其中的 pc 是前面定义的字符指针}

pc := pchar(integer(str));

showmessage(pc);{delphi}

end;

一个字符串(ansistring 或 string, 譬如是 "form1" )在内存中是这样储存的:

f

o

r

m

1

黄色区域是真正存字符串的位置, 前面说的字符串所在的内存地址, 就是本例中的 "f" 所在的位置;

蓝色的四个字节储存一个 integer 值, 表示字符串的长度;

最后红色的一个字节储存一个空字符(#0), 表示字符串的结束, 同时也是为了和 windows 的 null 结束的字符串兼容;

绿色的四个字节也是一个 integer 值, 表示该字符串被引用的次数(也就是有几个字符串的指针指向它).

还是看例子吧:

var

str,s1,s2: string;

pint: pinteger;

begin

str := self.text; {把窗体标题给它吧; 现在 str 指向了窗体标题所在的内存位置}

s1 := str;{给 s1 赋值}

s2 := str;{给 s2 赋值; 现在窗体标题已经有了 str、s1、s2 三个引用}

{str、s1、s2 的指针肯定不一样; 但现在指向内存的同一个位置, 测试:}

showmessage(inttostr(integer(str))); {15190384}

showmessage(inttostr(integer(s1)));{15190384}

showmessage(inttostr(integer(s2)));{15190384}

{向左偏移 4 个字节就是字符串长度的位置, 读出它来(肯定是5):}

pint := pinteger(integer(str) - 4);

showmessage(inttostr(pint^));{5}

{向左偏移 8 个字节就是字符串的引用计数, 读出它来(肯定是3):}

pint := pinteger(integer(str) - 8);

showmessage(inttostr(pint^));{3}

end;

当某段字符串内存的引用计数为 0 时, delphi 就会自动释放它; 这也是字符串不需要手动释放的原因.

我在测试时发现: 所有常量和非全局的变量的引用计数一直是 "-1".


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值