我有一个WPF应用程序,通过工具包(VS 2008)使用自动完成框。我有大约2000个记录的潜在人口,我试图通过结合填充事件程序来提高性能。我的结果不一致。过滤器似乎没问题,但我可以运行应用程序一次,结果X将在那里,但结果Y不会。再次运行它可以使结果Y在那里而不是X,随后X和Y都会在那里等,等等。这是我第一次使用自动完成框,所以我确定它必须是我的代码中的东西我忘记了。如果我在Itemsource绑定之前检查我的结果集,则会出现所需的结果,但用户无法看到它们 - 下拉自动完成功能不会显示。也许我需要一个事件覆盖???
XAML
<input:AutoCompleteBox
Name="autGlobal"
FilterMode="Contains"
Style="{DynamicResource MiniSearchAutoBoxWPF}"
IsTextCompletionEnabled="false"
Margin="5, 0, 5, 0"
HorizontalAlignment="Center"
KeyUp="autGlobal_KeyUp"
Text="Search Term"
GotFocus="autGlobal_GotFocus"
ValueMemberPath="Item"
Populating="AutoCompleteBox_Populating"
>
方法
private void AutoCompleteBox_Populating(object sender, PopulatingEventArgs e)
{
e.Cancel = true;
var b = new BackgroundWorker();
currSearch = autGlobal.Text;
b.DoWork += b_DoWork;
b.RunWorkerCompleted += b_RunWorkerCompleted;
b.RunWorkerAsync(autGlobal.Text);
}
private void b_DoWork(object sender, DoWorkEventArgs e)
{
Results.Clear();
int counter = 0;
string search = e.Argument.ToString();
search = search.ToUpper();
foreach (GlobalSearchList person in myGlobalList)
{
if (person.Item.ToUpper().Contains(search))
{
Results.Add(person);
counter++;
if (counter >= MAX_NUM_OF_RESULTS)
{
break;
}
}
}
}
private void b_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
if (this.Dispatcher.Thread == System.Threading.Thread.CurrentThread)
{
//Set the source
if (currSearch == autGlobal.Text)
{
autGlobal.ItemsSource = Results;
autGlobal.PopulateComplete();
}
}
else
{
this.Dispatcher.Invoke(new Action(() =>
{
//Set the source
if (currSearch == autGlobal.Text)
{
autGlobal.ItemsSource = Results;
autGlobal.PopulateComplete();
}
}));
}
}
答案 0 :(得分:3)
我不确定为什么你首先需要性能提升,你试图计算应该在另一个线程的自动完成框中的元素,然后将它们分配给ItemsSource
属性控制。类似的东西就是AutoCompleteBox应该做的事情。
我尝试将控件绑定到一个包含10000个字符串的列表,并且它工作正常,因此您的问题可能是您在集合中放置的对象的大小。一种解决方案可能只使用字符串表示,然后当您需要所选对象时,您可以根据它的表示找到它,假设它是唯一的(如果不是,您可以放置某种ID)。
这种方法的一个主要问题是线程sincronization,我现在将解释为什么你得到了外部行为,即使过滤器很好,结果中的项目也不对。
过滤器似乎没问题,但我可以运行应用程序一次,结果X会 在那里,但结果Y不会。再次运行可以使结果Y成为 那里而不是X,随后X和Y都会在那里等, 等
假设您在自动填充框中写入“ab”,这将启动执行此搜索的新BackGroundWorker
。如果你等待足够长的时间,一切都应该没问题。但是如果在第一个工作完成之前更改搜索查询,现在所有结果都将混合。例如,以下代码行:
// the user searchs for "ab"
[Thread 1] Results.Clear();
[Thread 1] Results.Add(Item[1]);
[Thread 1] Results.Add(Item[2]);
...
// the user changes the search term (to "abc" for example)
[Thread 2] Results.Clear();
[Thread 2] Results.Add(Item[3]);
// but what would happen if the first BackGroundWorker hasn't finished yet,
// this means that the first thread is still running
[Thread 1] Results.Add(Item[5]); // this items doesn't match the second search
[Thread 1] Results.Add(Item[6]); // criteria, but are added to the collection
[Thread 2] Results.Add(Item[7]);
// then you'll have two treads adding items to the Result collection
[Thread 1] Results.Add(Item[2]);
...
[Dispatcher Thread] autGlobal.ItemsSource = Results;
[Dispatcher Thread] autGlobal.PopulateComplete();
希望这有帮助。
答案 1 :(得分:0)
也许你可以查看这个。 为您完成的所有工作。