你真的需要重新设计你的 UI。在一个屏幕上使用具有相同值列表的512 个TComboBox
控件没有逻辑意义,而且浪费时间和资源。有更好的方法可以在屏幕上显示 512 个字符串,例如TListView
in report 模式或 a TListBox
(它们都支持虚拟模式,因此它们可以共享公共数据而不会浪费内存)。或者使用TValueListEditor
orTStringGrid
和esPickList
内联编辑器。或者,如果你真的很喜欢冒险,从头开始编写一个自定义控件,这样你就可以使用 1 个高效控件而不是 512 个单独的控件。任何东西都比 512TComboBox
控件好。
话虽这么说,TComboBox
不支持虚拟模式,like TListBox
and TListView
do,但是您仍然可以进行一些优化来加快您的TComboBox
es 速度:
不要制作相同TStringList
内容的 512 份副本。您添加到 的任何内容TComboBox::Items
都存储在TComboBox
的内存中。你应该努力重用你的单曲TStringList
,并让一切都根据需要委托给它。在这种情况下,您可以将TComboBox::Style
属性设置为csOwnerDrawFixed
并使用事件来按需TComboBox::OnDrawItem
绘制字符串。TStringList
您仍然需要向 each 添加字符串TComboBox
,但它们至少可以是空字符串。
子类TComboBox
覆盖其虚CreateParams()
方法并删除CBS_HASSTRINGS
窗口样式,则TComboBox
实际上不需要在其内存中存储空字符串。
尝试这样的事情:
class TMyComboBox : public Vcl::Stdctrls::TComboBox
{
typedef Vcl::Stdctrls::TComboBox inherited;
private:
TStrings *fSharedItems;
void __fastcall SetSharedItems(TStrings *Values)
{
if (fSharedItems != Values)
{
fSharedItems = Values;
Items->BeginUpdate();
try
{
Items->Clear();
if (fSharedItems)
{
for (int i = 0; i < fSharedItems->Count; ++i)
Items->Add(L"");
}
}
__finally
{
Items->EndUpdate();
}
}
}
protected:
virtual void __fastcall CreateParams(TCreateParams &Params)
{
inherited::CreateParams(Params);
Params.Style &= ~CBS_HASSTRINGS;
}
virtual __fastcall DrawItem(int Index, TRect Rect, TOwnerDrawState State)
{
// draw the items however you want...
if (fSharedItems)
Canvas->TextRect(Rect.Left, Rect.Top, fSharedItems->Strings[Index]);
}
public:
__fastcall TMyComboBox(TComponent *Owner)
: Vcl::Stdctrls::TComboBox(Owner)
{
Style = csOwnerDrawFixed;
}
__property TStrings* SharedItems = {read=fSharedItems, write=SetSharedItems};
};
class TMyForm : public TForm
{
...
private:
TStringList* targetlist;
TMyComboBox **com;
void __fastcall MyComboTriggerChange(TObject *Sender);
...
public:
__fastcall TMyForm(TComponent *Owner);
__fastcall ~TMyForm();
...
};
__fastcall TMyForm::TMyForm(TComponent *Owner)
: TForm(Owner)
{
targetlist = new TStringList;
targetlist->Add("Normal");
targetlist->Add("Inverted");
com = new TMyComboBox*[512];
for(int i=0;i<512;++i)
{
com[i] = new TMyComboBox(this);
com[i]->Parent = this;
com[i]->Name = String().sprintf(L"Combo_%d", i);
com[i]->SetBounds(10, 198 + 20 * i, 130, 200);
com[i]->SharedItems = targetlist;
com[i]->ItemIndex = 0;
com[i]->OnChange = &MyComboTriggerChange;
}
}
__fastcall TMyForm::~TMyForm()
{
delete targetlist;
delete[] com;
}
void __fastcall TMyForm::MyComboTriggerChange(TObject *Sender)
{
TMyComboBox *cb = static_cast<TMyComboBox*>(Sender);
// use targetlist->Strings[cb->ItemIndex] as needed...
}