Delphi中使用ListView和TreeView的Item中的Data可能被忽略的内存泄漏

el/2023/6/3 16:50:54

Delphi中大家在使用 ListView和TreeView时,一般都会使用到Item的Data属性来保存大家自己的内部数据,这个Data的声明如下:

property Data: Pointer;

大家可以看到它就是一个指针,使用上一般类似如下代码(以ListView为)

添加时

  PViewItemState = ^TViewItemState;
  TViewItemState = record
    ViewItemId: integer;
  end;

var
  nLoop: integer;
  li: TListItem;
  pp: PViewItemState;
begin
  for nLoop := 0 to 10 do
  begin
    li := ListView1.Items.Add;
    li.Caption := IntToStr(nLoop);
    new(pp);
    pp^.ViewItemId := nLoop;
    li.Data := pp;
  end;
end;


删除相应的项时多数采用

  ListView1.DeleteSelected;  或

  ListView1.Clear;

就结束了操作。

但是这里存在非常大的危险,就是我们的自己分配的Data内容有没有释放,Delphi文档中也没有关于这个问题的说明,所以我做了一个测试,发现Delphi是不自动释放这些数据的,这些未释放的数据在一个包含较多修改的ListView或者TreeView中将非常危险,如果是长期运行的系统将更危险,Delphi文档中也没有提示这些数据需要释放。

其实这个问题容易被大家忽略,解决办法还是有的,而且也不难,这里就要使用到ListView或者TreeView的OnDeletion事件,Delphi中关于OnDeletion的说明如下:

Occurs when an item in the list view is about to be deleted.

Delphi syntax:

property OnDeletion: TLVDeletedEvent;

C++ syntax:

__property TLVDeletedEvent OnDeletion = {read=FOnDeletion, write=FOnDeletion};

Description

Write an OnDeletion event handler to respond when an item in the list is about to be deleted. The Item parameter is the TListItem object from the Items property that is about to be deleted.

也就是说当需要删除项目时会触发这个事件,在VCL Source 中也可以看到调用过程,其实解决方法就是在这个事件中加入一个语句就可以了,并且是不可以缺少的语句,代码如下:

procedure TForm1.ListView1Deletion(Sender: TObject; Item: TListItem);
begin
  Dispose(Item.Data);
end;


这样就可以安全的释放掉Data的内存了。

http://www.ngui.cc/el/414491.html

相关文章

在CXGRID中显示'合计‘

设为:footerkind:sknone; procedure TFrmPOOrder.myviewTcxGridDBDataControllerTcxDataSummaryFooterSummaryItems4GetText( Sender: TcxDataSummaryItem; const AValue: Variant; AIsFooter: Boolean; var AText: String); begin Atext : 合计; end;

如何动态建立cxGrid的列?

for(int i0; i<ADOTable1->Fields->Count; i){ cxGrid1DBTableView1->CreateColumn(); cxGrid1DBTableView1->Columns[i]->DataBinding->FieldNameADOTable1->Fields->Fields[i]->FieldName; }

用友账套号修改及合并同账套多账套号多年度

用友账套号修改及合并同账套多账套号多年度 请仔细阅读代码后再小心使用 主要适用于将同一账套&#xff0c;单由于建账时分别使用了多账套号 例如&#xff1a;001-2005&#xff0c;002-2006&#xff0c;但确实是一个公司的帐&#xff0c;想合并成一个账套号&#xff0c;变成001…

去除存货的批次

/* 功能&#xff1a;去除存货的批次 步骤&#xff1a;在账套数据库里执行下列语句后进入软件整理现存量即可 */ --清除收发存记录的批次记录&#xff08;批次设置为空&#xff09; update rdrecords set cbatchnull --清除调拨记录的批次记录&#xff08;批次设置为空&#x…

调拨单等单据定位功能没有过滤条件

/*调拨单等单据定位功能没有过滤条件 ufdata_001_2009为有问题账套&#xff0c;ufdata_002_2009为新建账套 */ TRUNCATE TABLE ufdata_001_2009.dbo.rpt_fltdef TRUNCATE TABLE ufdata_001_2009.dbo.rpt_flddef TRUNCATE TABLE ufdata_001_2009.dbo.rpt_glbdef set IDENTIT…