我有一个ListView
,它将绑定到ObservableCollection
。当我手动删除项目(每个按钮事件)时,项目将被删除,ListView
立即刷新。
当我删除for循环中的项目时,在删除所有元素后,只有在操作结束时刷新列表。在项目被删除之前,对它们进行一些耗时的操作,并在另一个ListView
中通知用户。
那么如何才能实现,每次删除项目后列表都会刷新。
问题不在于,如何从列表中删除循环中的项目,因为这可以工作(按相反的顺序)。问题是,为什么在删除项目时UI不会刷新。
这里我简化了示例代码,它显示了这种行为:
XAML
<Window x:Class="ListViewTest.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:ListViewTest"
mc:Ignorable="d"
Title="MainWindow" Height="350" Width="525">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*"/>
</Grid.ColumnDefinitions>
<Grid.RowDefinitions>
<RowDefinition Height="*"/>
<RowDefinition Height="30"/>
<RowDefinition Height="30"/>
</Grid.RowDefinitions>
<ListView ItemsSource="{Binding LstFiles}" Grid.Column="0" Grid.Row="0"/>
<Button Content="Los" Name="losBtn" Click="losBtn_Click" Grid.Column="0" Grid.Row="1" />
<Button Content="del" Name="delBtn" Click="delBtn_Click" Grid.Column="0" Grid.Row="2" />
</Grid>
</Window>
C#落后
using System.Collections.ObjectModel;
using System.Windows;
namespace ListViewTest
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public ObservableCollection<string> LstFiles { get; set; }
public MainWindow()
{
InitializeComponent();
LstFiles = new ObservableCollection<string>();
for (int i = 0; i < 20; i++)
{
LstFiles.Add(string.Format("file_{0}", i));
}
DataContext = this;
}
private void losBtn_Click(object sender, RoutedEventArgs e)
{
for (int i = LstFiles.Count - 1; i >= 0; i--)
{
LstFiles.RemoveAt(i);
System.Threading.Thread.Sleep(50);
}
}
private void delBtn_Click(object sender, RoutedEventArgs e)
{
LstFiles.RemoveAt(0);
}
}
}
睡眠用于模拟耗时的操作。
为什么在删除项目时ui没有刷新
这是因为你正在调用Thread.Sleep()
,它只是阻止UI线程。您绝不能在UI应用程序中调用它。
虽然目前还不清楚为什么你想要这个“删除循环”(而不是仅仅清除ObservableCollection),你可以在Task.Delay()
点击处理程序方法中使用async
而不是Sleep:
private async void losBtn_Click(object sender, RoutedEventArgs e)
{
for (int i = LstFiles.Count - 1; i >= 0; i--)
{
LstFiles.RemoveAt(i);
await Task.Delay(50);
}
}
如果不是简单的延迟而是需要调用一些耗时的操作,您可以简单地将其包装在Task.Run
调用中:
private async void losBtn_Click(object sender, RoutedEventArgs e)
{
for (int i = LstFiles.Count - 1; i >= 0; i--)
{
LstFiles.RemoveAt(i);
await Task.Run(() =>
{
// some long running operation
});
}
}