Как запустить код внутри конвертера в отдельном потоке, чтобы пользовательский интерфейс не зависал? - PullRequest
14 голосов
/ 01 июля 2011

У меня медленный конвертер WPF (вычисления, онлайн-выборка и т. Д.).Как я могу преобразовать асинхронно, чтобы мой пользовательский интерфейс не зависал?Я нашел это, но решение состоит в том, чтобы поместить код конвертера в свойство - http://social.msdn.microsoft.com/Forums/pl-PL/wpf/thread/50d288a2-eadc-4ed6-a9d3-6e249036cb71 - что я бы предпочел не делать.

Ниже приведен пример, демонстрирующий проблему.Здесь выпадающий список будет зависать до тех пор, пока не пройдет сон.

namespace testAsync
{
    using System;
    using System.Collections.Generic;
    using System.Threading;
    using System.Windows;
    using System.Windows.Data;
    using System.Windows.Threading;

    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            MyNumbers = new Dictionary<string, int> { { "Uno", 1 }, { "Dos", 2 }, { "Tres", 3 } };

            this.DataContext = this;           
        }

        public Dictionary<string, int> MyNumbers
        {
            get { return (Dictionary<string, int>)GetValue(MyNumbersProperty); }
            set { SetValue(MyNumbersProperty, value); }
        }
        public static readonly DependencyProperty MyNumbersProperty =
            DependencyProperty.Register("MyNumbers", typeof(Dictionary<string, int>), typeof(MainWindow), new UIPropertyMetadata(null));


        public string MyNumber
        {
            get { return (string)GetValue(MyNumberProperty); }
            set { SetValue(MyNumberProperty, value); }
        }
        public static readonly DependencyProperty MyNumberProperty = DependencyProperty.Register(
            "MyNumber", typeof(string), typeof(MainWindow), new UIPropertyMetadata("Uno"));
    }

    public class AsyncConverter : IMultiValueConverter
    {
        public object Convert(object[] values, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        {
            object result = null;


            if (values[0] is string && values[1] is IDictionary<string, int>)
            {
                DoAsync(
                    () =>
                        {
                                            Thread.Sleep(2000); // Simulate long task
                            var number = (string)(values[0]);
                            var numbers = (IDictionary<string, int>)(values[1]);

                            result = numbers[number];
                            result = result.ToString();
                        });
            }

            return result;
        }

        private void DoAsync(Action action)
        {
            var frame = new DispatcherFrame();
            new Thread((ThreadStart)(() =>
            {
                action();
                frame.Continue = false;
            })).Start();
            Dispatcher.PushFrame(frame);
        }

        public object[] ConvertBack(object value, Type[] targetType, object parameter, System.Globalization.CultureInfo culture)
        {
            throw new NotSupportedException();
        }
    }

и XAML:

<Window x:Class="testAsync.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:testAsync"
        Title="MainWindow" Height="200" Width="200">
    <Window.Resources>
        <local:AsyncConverter x:Key="asyncConverter"/>
    </Window.Resources>
    <DockPanel>
        <ComboBox DockPanel.Dock="Top" SelectedItem="{Binding MyNumber, IsAsync=True}"                   
                  ItemsSource="{Binding MyNumbers.Keys, IsAsync=True}"/>
        <TextBlock DataContext="{Binding IsAsync=True}"
            FontSize="50" FontWeight="Bold" HorizontalAlignment="Center" VerticalAlignment="Center">
            <TextBlock.Text>
                <MultiBinding Converter="{StaticResource asyncConverter}">
                    <Binding Path="MyNumber" IsAsync="True"/>
                    <Binding Path="MyNumbers" IsAsync="True"/>
                </MultiBinding>
            </TextBlock.Text>
        </TextBlock>
    </DockPanel>
</Window>

Обратите внимание, что все привязки теперь IsAsync = "True", но это не помогает.

enter image description here

Поле со списком застрянет на 2000 мс.

Ответы [ 4 ]

6 голосов
/ 01 июля 2011

Я знаю, что вы сказали, что не хотите вызывать перевод из установщика свойств, но я утверждаю, что это более чистый подход, чем IValueConverter / IMultiValueConverter.

В конечном итоге вы хотите установить значение выбранного числа из выпадающего списка и сразу же вернуться к нему. Вы хотите отложить обновление отображаемого / переведенного значения до завершения процесса перевода.

Я думаю, что яснее моделировать данные так, чтобы преобразованное значение само было свойством, которое просто обновляется асинхронным процессом.

    <ComboBox SelectedItem="{Binding SelectedNumber, Mode=OneWayToSource}"                   
              ItemsSource="{Binding MyNumbers.Keys}"/>
    <TextBlock Text="{Binding MyNumberValue}" />

public partial class MainWindow : Window, INotifyPropertyChanged
{
    public MainWindow()
    {
        InitializeComponent();

        MyNumbers = new Dictionary<string, int> { { "Uno", 1 }, { "Dos", 2 }, { "Tres", 3 } };

        DataContext = this;   
    }

    public IDictionary<string, int> MyNumbers { get; set; }

    string _selectedNumber;
    public string SelectedNumber
    {
        get { return _selectedNumber; }
        set
        {
            _selectedNumber = value;
            Notify("SelectedNumber");
            UpdateMyNumberValue();
        }
    }

    int _myNumberValue;
    public int MyNumberValue
    {
        get { return _myNumberValue; }
        set 
        { 
            _myNumberValue = value;
            Notify("MyNumberValue");
        }
    }

    void UpdateMyNumberValue()
    {
        var key = SelectedNumber;
        if (key == null || !MyNumbers.ContainsKey(key)) return;

        new Thread(() =>
        {
            Thread.Sleep(3000);
            MyNumberValue = MyNumbers[key];
        }).Start();
    }

    public event PropertyChangedEventHandler PropertyChanged;
    void Notify(string property)
    {
        var handler = PropertyChanged;
        if(handler != null) handler(this, new PropertyChangedEventArgs(property));
    }
}
5 голосов
/ 01 июля 2011

Вы можете использовать для этого DispatcherFrame, вот пример конвертера:

public class AsyncConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        object result = null;
        DoAsync(() =>
        {
            Thread.Sleep(2000); // Simulate long task
            result = (int)value * 2; // Some sample conversion
        });
        return result;
    }

    private void DoAsync(Action action)
    {
        var frame = new DispatcherFrame();
        new Thread((ThreadStart)(() =>
        {
            action();
            frame.Continue = false;
        })).Start();
        Dispatcher.PushFrame(frame);
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        throw new NotSupportedException();
    }
}
1 голос
/ 01 июля 2011

Не стоит создавать тяжелые вычисления в конвертере, особенно если вы выполняете функциональность, которую другие должны использовать в качестве хорошего примера.

Я бы переписал и использовал MVVM с вашей ViewModel в качестве конвертера на стероидах, где вы можете делать все эти вещи прозрачным способом - проще в программировании, более понятном потоке программ, легче в понимании кода.

И тогда вы можете использовать Приоритетные привязки:

http://msdn.microsoft.com/en-us/library/system.windows.data.prioritybinding.aspx

Для вашей первоначальной задачи я бы посмотрел, когда вызывается конвертер - если это происходит, когда привязка вернула свое значение, вы, вероятно, можетезаставить Async делать то, что он делает.Я подозреваю, что wpf ожидает возврата свойства и затем вызывает конвертер - в этом случае может оказаться невозможным заставить ваш конвертер не заморозить графический интерфейс.

Подход, который вы могли бы предпринять:

  • В вашем конвертере вы должны начать извлекать ваши данные и возвращаться, например, с помощью backgroundworker - в противном случае пользовательский интерфейс будет зависать.
  • В мультибиндинге передайте ссылку на что-то, чтобы при поступлении ваших данных вы могли запускать propertyloaded
0 голосов
/ 01 июля 2011

Предлагаю посмотреть на BackgroundWorker. Он может выполнить перевод в фоновом потоке, а затем вызвать завершенное событие в потоке пользовательского интерфейса.

См. http://www.dotnetperls.com/backgroundworker

...