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

  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的内存了。