在Firemonkey中创建然后销毁运行时的TLabels



我正在尝试在运行时生成 TLabels 并使用此代码将它们插入到 VertScrollBox 中;

var
   i, f: integer;
   RT_Label: TLabel;
begin
   f:= 10;
   for i := 0 to 20 do
   begin
        RT_Label := TLabel.Create(Self);
        RT_Label.Name := 'Label' + i.ToString;
        RT_Label.Text := 'SampleLabel' + i.ToString;
        RT_Label.Position.Y := f;
        RT_Label.Align := TAlignLayout.Top;
        RT_Label.Parent := VertScrollBox1;
        inc(f, 15);
   end;
end; 

标签显示没有任何问题,但是当我尝试使用此代码释放生成的标签时:

var
   i: integer;
   LComponent: TComponent;
begin
   for i := 0 to ComponentCount-1 do
   begin
        if( Components[i] is TLabel )then
         if StartsText('Label', (Components[i] as TLabel).Name) then
         begin
             LComponent := (Components[i] as TLabel);     
             If Assigned(LComponent) then FreeAndNil(LComponent);
         end;
    end;
end;

然后我总是收到错误"参数超出范围"。

如何正确删除在运行时添加到 VertScrollBox 的 TLabels?

从以下行开始循环

for i := 0 to ComponentCount-1 do

但是,当您释放组件时,它会将自身作为其清理代码的一部分从组件列表中删除。因此,每个释放的组件都会将列表的大小减小 1。当 for 循环启动时,ComponentCount-1表达式将计算一次,因此不会更新以反映更改。

即使你可以解决这个问题,你的循环也会跳过项目。即,如果您删除了第 3 项,则第 4 项现在将成为第 3 项,但您的循环将前进到第 4 项。

不过,解决这个问题的方法很简单。只需向后迭代列表:

for i := ComponentCount-1 downto 0 do

值得一提的是,您的代码实际上只会释放Windows和OSX上的项目。在移动设备上,编译器使用 ARC,它仅在删除所有引用后释放对象。解决方案/解决方法/fudge[1]是调用DisposeOf而不是组件的Free

顺便说一句,as算子已经保证了对象是Assigned的,所以不需要额外的测试。无需FreeAndNil将重新赋值或直接超出范围的局部变量,也无需在释放对象之前强制转换对象。由于 Free(或 DisposeOf)方法存在于公共祖先类中,编译器将解析任何后代类的链接。

因此,您的代码可以简化为:

var
  i: integer;
begin
  for i := ComponentCount-1 downto 0 do
  begin
    if Components[i] is TLabel then
      if StartsText('Label', (Components[i] as TLabel).Name) then
        Components[i].DisposeOf;
  end;
end;

[1] - 取决于您与谁交谈。

相关内容

  • 没有找到相关文章

最新更新