`
ivfh
  • 浏览: 60467 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
最近访客 更多访客>>
社区版块
存档分类
最新评论

Delphi中树型控件的使用技巧

阅读更多

 我们都知道,开发者主要用Delphi来开发数据库管理软件,正因如此,树型控件的使用最好与数据库联系起来。Delphi提供了一个树型控件TTreeView,可以用来描述复杂的层次关系。

树节点信息的存储和加载

常用的方法是用树控件的 LoadFromFile和SavetoFile方法,来实现树控件和文件之间的交互;或用Assign方法实现树控件和DBMemo,也就是和数据库间的交互。该方法的优点是编程相对简单,缺点是树控件的实际节点数可能会很大,对于“大树”,每次加载和存储的数据量会加大,将降低速度,增大系统开销,造成数据冗余。另一种方法,就是只在树上产生“看得见”的节点,没有专门记录全部树节点结构的文件或数据库字段,而将树节点结构分散在数据库的每一个记录中。

具体方法是:创建一个数据库,字段根据实际业务而定,其中必然有一个字段的信息将在树型控件的节点上显示,另外还要一个字段来保存节点的惟一标识号,该标识号由长度相等的两部分组成,前段表示当前节点的父节点号,后段表示当前节点的节点号,此标识号相当于一个“链表”,记录了树上节点的结构。该方法的优点:用户操作“大树”时,一般不会展开所有的节点,而只用到有限的一部分,同时只能从树根一层一层地展开,该法只在树上产生“看得见”的节点,所以,存储和加载“大树”的速度快,数据量小,系统开销和数据冗余较小。缺点:编程较复杂,但可以结合该方法编成一个新的树控件,将大大提高编程效率。值得注意的是,ID号必须惟一,所以在编程中如何合理产生ID尤为重要。

数据库结构示例

创建一个数据库,为简化程序,我只创建两个数据库字段,定义如下:

字段名 类型 长度

text c 10

longid c 6

LongID字段实际上由两段组成,每一段3位,LongID只能表示1000条记录。将LongID定义为索引字段,存为c:\testtree\tree.dbf。编辑该DBF文件,新建一条记录,Text字段设为TOP,LongID字段设为“000”(3个“0”前为三个空格)。

创建演示程序

在Form1上放置TreeView1、Table1、PopupMenu1、Edit1、Edit2。TreeView1的PopupMenu属性设为PopupMenu1;Table1的DataBaseName属性设为c:\testtree,TableName属性设为tree.dbf,IndexFieldNames属性设为LongID;为PopupMenu1加选单项Add1和Del1,Caption分别为Add和Del;Edit1用来输入新节点的Text属性值,Edit2用来输入新节点的3位ID号。存为c:\testtree\treeunit.pas和c:\testtree\testtree.dpr。

在treeunit.pas的Type关键字后加入一行:Pstr=^string;{Pstr为字符串指针}

为Form1的OnCreate事件添加代码:

procedure TForm1.FormCreate(Sender: TObject);

var p:Pstr;Node:TTreeNode;

begin

with Table1,Treeview1 do


begin

open;

first;

new(p);{为指针p分配内存}

p^:=FieldByName(′LongID′).AsString;

Node:=Items.AddChildObject(nil,FieldByName(′Text′).AsString,p);

if HasSubInDbf(Node) then Items.AddChildObject(Node,′ ′,nil);{有子节点则加一个空子节点}

end;

end;

 

HasSubInDbf为自定义函数,自变量为Node,检查节点Node有无子节点,有则返回True,反之返回False,并在TForm1的类定义里加入原型声明(其它自定义函数的原型也在TForm1的类定义里声明,不另作解释),函数代码如下:

function TForm1.HasSubInDbf(Node:TTreeNode):Boolean;

begin

with Table1 do

begin

Table1.FindNearest([copy(Pstr(Node.Data)^,4,3)+′000′]);

result:=copy(FieldByName(′LongID′).AsString,1,3)=copy(Pstr(Node.Data)^,4,3);{如数据库里当前记录的LongID字段内容的前3位和节点Node的Data的后3位相同,则Node应该有子节点}

end;

end;

  为TreeView1控件的OnDeletion事件添加代码,需要指出的是,不仅调用Delete方法可以触发OnDeletion事件,而且当树控件本身被释放前,也触发OnDeletion事件,所以,在此处加入dispose(node.data)会很“安全”:

procedure TForm1.TreeView1Deletion(Sender: TObject; Node: TTreeNode);

begin

Dispose(Node.Data);{释放节点数据内存}

end;

 为Add1选单项的OnClick事件添加代码如下:

procedure TForm1.Add1Click(Sender: TObject);

var p:pstr;Tmpstr:string;i:integer;

begin

try

StrToInt(Edit2.Text);

Tmpstr:=Edit2.Text;{注:在实用中,必须用更好的方法来产生ID}

except;

ShowMessage(′重新输入Edit2的内容′);

abort;

end;

with TreeView1 do

begin

new(p);

p^:=copy(Pstr(Selected.Data)^,4,3)+TmpStr;

Items.AddChildObject(Selected,Edit1.Text,p);


end;

with Table1 do{ 在数据库里添加记录 }

begin

Append;

FieldByName(′Text′).AsString:=Edit1.text;

FieldByName(′LongID′).AsString:=p^;

Post;

end;

TmpStr:=inttostr(strtoint(TmpStr)+1);

for i:=length(TmpStr) to 2 do TmpStr:=′0′+TmpStr;

Edit2.Text:=TmpStr;

end;

 为Del1菜单项的OnClick事件添加代码如下:

procedure TForm1.Del1Click(Sender: TObject);

var DelList:TStringList;LongID,NSubLongID:string;

begin

DelList:=TStringList.create;

DelList.Sorted:=True;

DelList.Add(Pstr(TreeView1.Selected.Data)^);

while DelList.Count>0 do

begin

LongID:=DelList.Strings[0];

DelList.Delete(0);

Table1.SetKey;

Table1.FieldByName(′LongID′).AsString:=LongID;

if Table1.GotoKey then Table1.Delete;

if HasSubInDbf(TreeView1.Selected) then

begin

NSubLongID:=Table1.FieldByName(′LongID′).AsString;

while (copy(NSubLongID,1,3)=copy(LongID,4,3))and(not Table1.Eof) do

begin

dellist.Add(NSubLongId);

Table1.Next;

NSubLongId:=Table1.FieldByName(′LongID′).AsString;

end;

end;

end;

DelList.Free;

TreeView1.Items.Delete(TreeView1.Selected);

end;

为TreeView1的OnEXPanding事件添加代码:

procedure TForm1.TreeView1EXPanding(Sender: TObject; Node: TTreeNode;

var AllowEXPansion: Boolean);

var TmpNode:TTreeNode;NSubLongID:String;p:Pstr;bm:TBookMark;


begin

with Table1,TreeView1 do

begin

Items.BeginUpdate;

SetKey;

FieldByName(′LongID′).AsString:=Pstr(Node.Data)^;

if not GotoKey then Items.Delete(Node)

else

begin

TmpNode:=Node.GetFirstChild;

if (TmpNode.Text=′ ′)and(TmpNode.Data=nil) then

begin

TmpNode.Delete;

if HasSubInDbf(Node) then

begin

NSubLongID:=FieldByName(′LongID′).AsString;

while (copy(NSubLongID,1,3)=copy(Pstr(Node.Data)^,4,3))and(not Eof) do

begin

new(p);

p^:=FieldByName(′LongID′).AsString;

bm:=GetBookMark;

TmpNode:=Items.AddChildObject(Node,FieldByName(′Text′).AsString,p);

if HasSubInDbf(TmpNode) then Items.AddChildObject(TmpNode,′ ′,nil);

GotoBookMark(bm);

FreeBookMark(bm);

Next;

NSubLongId:=FieldByName(′LongID′).AsString;

end; end; end;

end;

Items.EndUpdate;

end;

end;

 以上简要谈了谈数据库的树状显示的基本方法,另外,编辑树上节点的Text属性的同时对数据库进行修改、同一数据库在多用户同时操作时数据库以及树的一致性、树上节点的拷贝与复制等就不再赘述,读者可自行完善。本文程序在Dlphi4.0、Windows 98下调试通过

分享到:
评论

相关推荐

    delphi 树型控件自动根据数据集生成树型结构

    本文将详细讲解如何在 Delphi7 中使用树型控件,并自动根据数据集生成树型结构。 首先,理解 TTreeView 控件的基本操作。TTreeView 提供了一个可视化的组件,用于展示具有父节点和子节点的关系的数据。每个节点表示...

    delphi自定义控件_treeview控件加强

    在Delphi编程环境中,开发人员经常需要对标准控件进行扩展和定制,以满足特定项目的需求。本主题将深入探讨如何对Delphi的内置TreeView控件进行增强,以实现更丰富的功能,例如增删改及拖拽操作。我们将以一个名为`...

    Virtual-TreeView(树型控件)

    在 Delphi 中,虽然提供了 `TreeView` 和 `ListView` 控件,但它们的功能相对单一,往往无法满足一些特定的个性化需求。幸运的是,借助 **Virtual Treeview** 控件,我们可以轻松实现各种创新的功能。

    delphi里树形控件

    在Delphi编程环境中,树形控件是一种常用的用户界面元素,用于展示层次化的数据结构。标题中的"mydbtreeview"是特指一个自定义的、带有数据库存取功能的树形控件,它简化了在Delphi应用中处理树形数据与数据库之间的...

    delphi的TreeView的树型结构

    6. **使用技巧**: - **动态加载数据**:为了提高性能,可以在需要时异步加载子节点,如在节点展开时加载其子节点。 - **多选模式**:通过设置`SelectionMode`属性,可以启用多选模式,让用户同时选择多个节点。 ...

    Delphi实现树型结构具体实例

    在Delphi编程中,树型结构(TreeView)是一种常见的用户界面元素,用于展示层次化的数据。本实例将探讨如何在Delphi中实现一个具体的树型结构,并从数据源(如XML文件)加载数据。以下是对这个实例的详细解析: 1. ...

    几个delphi控件具体看介绍

    几个delphi控件,包含:TntUnicodeControls_2.3.0.zip-Unicode编码控件 THTMLWriter20120101-155121.zip-HTML树型编辑 BpJavaScript20101013.rar-js脚本控件 HtmlViewer-HtmlViewer-11.5.zip-html显示 foreval(8.4....

    Delphi 动态树结合数据库查询示例

    在本示例中,我们将探讨如何使用第三方控件RzCheckTree来创建一个动态的、与数据库关联的多级树型菜单。RzCheckTree是Rad Software Engineering公司开发的一个强大的树形控件,它提供了丰富的功能,如复选框支持、...

    Delphi读取数据库内容显示在树形视图中..rar

    在这个场景中,树形视图(TreeView)控件是一个常用的选择,因为它能清晰地组织和展示层次结构数据。本篇文章将深入探讨如何在Delphi中实现从数据库读取数据并显示在树形视图中的过程。 首先,我们需要了解Delphi中...

    用delphi写的显示所有进程信息树型结构

    在这个特定的案例中,我们有一个用Delphi编写的程序,它能够展示系统中所有进程的信息,并以树型结构呈现,包括各个进程的属性和资源占用情况。 首先,让我们深入了解一下"进程"这个概念。在操作系统中,进程是程序...

    TreeComboBox.v2.3.2.For.Delphi5679.CR

    在Delphi编程环境中,开发者经常需要处理复杂的数据展示问题,其中一种常见的需求就是在下拉框(ComboBox)中实现树型结构的显示。针对这一需求,"TreeComboBox.v2.3.2"控件应运而生,它为Delphi5、6、7及9的用户...

    delphi 界面自定义

    5.对象列表:与Delphi树型列表一致。 6.设计窗:指要调整的界面。在设计窗选中任意个控件按“Shift+Ctrl+方向键”时可以将所选控件大步移动,按“Ctrl+方向键”时可以将所选控年小步移动,按“Shift+方向键”时可以...

    delphi树形结构地域管理

    在“delphi树形结构地域管理”这个主题中,我们将深入探讨如何利用Delphi的组件和编程技巧来实现一个用于管理地域信息的树形控件。以下是一些关键知识点: 1. **TTreeView组件**:Delphi中用于显示树形结构的主要...

    delphi EhLib_v4.1.4k

    在Delphi编程环境中,扩展功能和增强应用性能往往离不开各种控件的支持。EhLib_v4.1.4k便是这样一个强大的组件库,专为Delphi开发者设计,旨在提供更丰富的图形用户界面(GUI)元素和更高效的数据库操作能力。本文将...

    Cxgrid做树形

    1. **安装和配置CxGrid**:首先,你需要在你的开发环境中(如Visual Studio)安装DevExpress套件,其中包含了CxGrid控件。安装完成后,可以在工具箱中找到并添加CxGrid到你的窗体上。 2. **创建数据源**:树形结构...

    DEPHI 树形控件

    在 Delphi 编程环境中,树形控件(TTreeView)是一种常见的用户界面元素,用于显示层次结构的数据。它在各种应用中都有广泛的应用,如文件系统浏览、菜单结构展示等。下面我们将深入探讨 Delphi 中如何操作树形控件...

    DevExpress.vcl.v45.rar_ExpressPrinting_delphi 报表_devExpress报表_de

    DevExpress公司出品的Borland Delphi和C++ Builder的控件(包含完整源代码). ExpressVerticalGrid:就是以前大家熟知的ExpressInspector。用于 Delphi 和 C++Builder的“反向”网格控制组件。 ExpressBar:菜单控件以及...

    由特定数据库表格式(3个字段)生成TreeView树型

    本项目似乎涉及到一个使用Delphi编程语言开发的应用,其目标是根据特定的数据库表(包含3个字段)生成一个TreeView控件,以展示层级关系。以下是关于这个主题的详细知识: 1. **TreeView控件**:TreeView控件在...

    滴答表格控件免费版3.6.3

    10.打印功能 支持直接在本控件中预览,无需弹出窗口; 支持页面设置; 支持设置打印机,纸张,页边 距,页面方向,份数,双面打印等等 ;支持页眉页脚设置; 支持打印固定行列,网格虚线; 11.文件存储功能 具有滴答表格自己的...

    DELPHI办公全套管理系统(无控件)

    偶为公司作的DELPHI版本OA办公系统,包括人事档案,工资,调动管理,采用树型结构!包括办公物资进出管理,办公各种工具包,涉及多个方面!完全开发源码,请大家参考指点!有什么意见请和我联系!saymycareer@sohu....

Global site tag (gtag.js) - Google Analytics