文档章节

分析 TStringList 存取对象的特点 - 还是回复 Test1234 的问题

涂孟超
 涂孟超
发布于 2014/09/26 15:38
字数 989
阅读 2
收藏 0
问题来源: http://www.cnblogs.com/del/archive/2008/04/26/973346.html#1171927

在 Delphi 中存储系列对象, 大家常用 TList 类; 有了 TObjectList(在 Contnrs 单元)以后, 存储对象就有了更好的选择, 因为从 TObjectList 列表中移除的对象同时会得到释放.

很少有人使用 TStringList 储存对象, 殊不知用 TStringList 储存对象也有 TList 和 TObjectList 所不及的优势.

我想在继续探讨前先重复一个概念: 对象的 "指针" 和 "首地址":
我们通过对象的指针可以找到对象, 也就是说指针是指向了对象; 对象也不过是一系列数据, "指针" 一般是指向这组数据的 "首地址".
下面代码可以获取 Button1 对象的 "指针" 和 "首地址":
procedure TForm1.Button1Click(Sender: TObject);
begin
  ShowMessageFmt('指针: %d', [Integer(@Button1)]);  {14910416}
  ShowMessageFmt('首地址: %d', [Integer(Button1)]); {15011440}
end;

 
 
 
 
 

 

 

  
我们再看下 TList、TObjectList 和 TStringList 添加对象的方法声明:
TList       : function Add(Item: Pointer): Integer;
TObjectList : function Add(AObject: TObject): Integer;
TStringList : function AddObject(const S: string; AObject: TObject): Integer;

 
 
 
 
 

 

 

  
可以看出, TList 添加的只是指针; TObjectList 和 TStringList 添加的类型是对象.
添加对象时, 是把整个对象的数据都添加进去吗? 当然不是, 只要记住对象的首地址就可以了(应该也是用类似指针的办法, 我没仔细研究), 测试代码:
procedure TForm1.Button1Click(Sender: TObject);
var
  List: TStringList;
begin
  ShowMessageFmt('指针: %d', [Integer(@Button1)]);  {14910416}
  ShowMessageFmt('首地址: %d', [Integer(Button1)]); {15011440}

  List := TStringList.Create;
  List.AddObject('btn', Button1);
  ShowMessageFmt('取出: %d', [Integer(List.Objects[0])]); {15011440, 可以看出相同与上面的首地址}
  List.Free;
end;

 
 
 
 
 

 

 

  
通过 TStringList 的 AddObject 和 InsertObject 方法可以添加对象;
用 Objects[] 属性取出对象; 用 List[] 取出字符串. 示例:
procedure TForm1.Button1Click(Sender: TObject);
var
  List: TStringList;
  obj: TObject;
  str: string;
begin
  List := TStringList.Create;
  List.AddObject('btn', Button1);    {添加}

  obj := List.Objects[0];            {取出对象}
  str := List[0];                    {取出字串}

  {使用对象, 有个前提:我们知道它属于 TButton}
  ShowMessage(TButton(obj).Caption); {Button1}
  ShowMessage(str);                  {btn}

  List.Free;
end;

 
 
 
 
 

 

 

  
添加对象的指针可以吗? 可以, 但需要转换成无类型指针, 例:
procedure TForm1.Button1Click(Sender: TObject);
var
  List: TStringList;
  obj: TObject;
begin
  List := TStringList.Create;
  List.AddObject('btn', Pointer(@Button1));
  List.AddObject('btn', Pointer(Button1)); {这样也可以}
  obj := List.Objects[0];
  ShowMessage(TButton(obj).Caption); {显示: Button1}
  List.Free;
end;

 
 
 
 
 

 

 

  
既然也可以添加指针, 那我们也可以添加不属于 TObject 的结构等其他指针;
假如不能添加指针, 也将无法添加结构, 因为结构不属于 TObject. 举例:
type
  PMyRec = ^TMyrec;
  TMyRec = record
    s: string;
    i: Integer;
  end;

procedure TForm1.Button1Click(Sender: TObject);
var
  List: TStringList;
  R1,R2: TMyRec;
begin
  List := TStringList.Create;

  R1.s := 'abc';
  R1.i := 123;
  List.AddObject('rec', Pointer(@R1)); 
  //List.AddObject('rec', @R1); {结构比较特殊, 不转无类型指针也可以}

  R2 := PMyRec(List.Objects[0])^;
  ShowMessageFmt('%s,%d', [R2.s, R2.i]); {abc,123}

  List.Free;
end;

 
 
 
 
 

 

 

  
前面说到 TStringList 还会有点优势; 首先得承认它的劣势, 因为它是两组数据构成的列表, 在数据量特别大的时候效率上会有劣势; 现在说说它的优势:
从 TList 和 TObjectList 取出的对象类型是未知的(当然作者知道), 所以一般只能存储单一类型的对象;
因为 TStringList 有两个字段, 我们可以用那个 String 字段来储存对象类型, 从而让 TStringList 可以同时储存更多类型的对象. 举例:
unit Unit1;

interface

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

type
  TForm1 = class(TForm)
    Button1: TButton;
    procedure Button1Click(Sender: TObject);
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}
type
  PMyRec = ^TMyrec;
  TMyRec = record
    s: string;
    i: Integer;
  end;

procedure TForm1.Button1Click(Sender: TObject);
var
  List: TStringList;
  R1,R2: TMyRec;
  str: string;
  i: Integer;
begin
  List := TStringList.Create;

  R1.s := 'abc';
  R1.i := 123;
  str := '我是字符串';
  List.AddObject('1', @R1);           {用 1 表示结构 TMyRec}
  List.AddObject('2', Sender);        {用 2 表示 TButton}
  List.AddObject('3', Self);          {用 3 表示 TForm1}
  List.AddObject('4', Pointer(str));  {用 4 表示 String}

  for i := 0 to List.Count - 1 do
  begin
    case StrToIntDef(List[i], 0) of
      1: begin
           R2 := PMyRec(List.Objects[i])^;
           ShowMessageFmt('%s,%d', [R2.s, R2.i]);       {abc,123}
         end;
      2: ShowMessage(TButton(List.Objects[i]).Caption); {Button1}
      3: ShowMessage(TForm1(List.Objects[i]).Text);     {Form1}
      4: ShowMessage(PChar(List.Objects[i]));           {我是字符串}
    end;
  end;

  List.Free;
end;

end.

 
 
 
 
 

 

 

  

本文转载自:http://www.cnblogs.com/del/archive/2008/04/26/1172589.html

共有 人打赏支持
涂孟超
粉丝 12
博文 2011
码字总数 14107
作品 0
深圳
程序员
objective-c 关于 self 的用法总结

今天新同事问了一些关于什么时候用全局变量,什么时候用self.赋值的问题,所以笔者在此说明一下。 何时使用self.在网上搜索或者论坛里的回复大多都是简简单单的说这与objc的存取方法有关,如...

鉴客
2011/12/14
7.8K
2
LINUX类主机JAVA应用程序占用CPU、内存过高分析手段

转载声明:本文为DBA+社群原创文章,转载必须连同本订阅号二维码全文转载,并注明作者名字及来源:DBA+社群(dbaplus)。 做为一个IT运维人员,通常在运维过程中会遇到各种各样的问题,系统问...

丁启良
2015/12/04
0
0
聚合(根)、实体、值对象精炼思考总结

聚合根、实体、值对象的区别? 从标识的角度: 聚合根具有全局的唯一标识,而实体只有在聚合内部有唯一的本地标识,值对象没有唯一标识,不存在这个值对象或那个值对象的说法; 从是否只读的...

大糊涂
2015/06/11
0
0
.NET中MD5编码的内存泄露问题分析

问题描述与定位 最近一个项目中要加工处理700多万条的三元组数据,总是在执行到二三百万条的时候就报内存溢出了。不断的检查代码,各种对象局部化;使用.net profiler分析堆栈内存,发现有大...

caltrop
2015/12/29
2.3K
14
数据库系统概述

什么是数据库 数据库(Database DB)指按照数据结构来组织,存储和管理数据的仓库,是存储在一起的相关数据的集合, 什么是数据库管理系统 数据库操作系统(Database Management System ,DBMS)是一...

乌龟漫步
2016/12/20
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

内存模型是怎么解决缓存一致性的?

在再有人问你Java内存模型是什么,就把这篇文章发给他。这篇文章中,我们介绍过关于Java内容模型的来龙去脉。 我们在文章中提到过,由于CPU和主存的处理速度上存在一定差别,为了匹配这种差距...

Java填坑之路
12分钟前
1
0
vue-cli 3.0 初体验

最近复习了下vue,突然发现vue-cli已经更新到3.0版本了,并且变化蛮大,看来要不停的学习,真是一入前端深似海。 安装步骤: 1、全局安装 npm install -g @vue/cli Vue CLI 的包名称由 vue-...

tianyawhl
14分钟前
0
0
Angular进阶之路

【初级】会写页面,能出东西。 给定环境和 rest API,不用第三方库,能在十分钟内完成一个 master/detail 结构的带路由的应用(可以不管美观)。 知识点:Angular CLI、组件、路由、HTTP 服务...

陆小七的主页
16分钟前
0
0
Redis缓存数据库安全加固指导(一)

背景 在众多开源缓存技术中,Redis无疑是目前功能最为强大,应用最多的缓存技术之一,参考2018年国外数据库技术权威网站DB-Engines关于key-value数据库流行度排名,Redis暂列第一位,但是原生...

中间件小哥
16分钟前
0
0
百万级数据mysql分区

1. 什么是表分区? 表分区,是指根据一定规则,将数据库中的一张表分解成多个更小的,容易管理的部分。从逻辑上看,只有一张表,但是底层却是由多个物理分区组成。 2. 表分区与分表的区别 分表...

罗文浩
19分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部