2012-02-20 28 views
5

私はRxライブラリでハンドルを取得しようとしていて、MVVMでWPFで処理しています。私はアプリケーションをリポジトリやViewModelなどのコンポーネントに分割しました。私のリポジトリは、学生のコレクションを1つずつ提供することができますが、ビューバインドObservableCollectionに追加しようとすると、スレッドエラーがスローされます。私はこれを私のために働かせるためにいくつかのポインタを謝るつもりです。別のスレッドからObservableCollectionを更新する

+0

可能重複(http://stackoverflow.com/questions/2104614/updating-an-observablecollection-in-a-separate-thread ) –

+0

任意のスレッドから動作し、複数のUIスレッドを介してバインドできるスレッドセーフソリューションを提供する次のリンクを試してください。http://www.codeproject.com/Articles/64936/Multithreaded-ObservableImmutableCollection – Anthony

答えて

8

あなたは正しく例えば

ObserveOn(SynchronizationContext.Current) 

このブログ記事を参照してください

http://10rem.net/blog/2011/02/17/asynchronous-web-and-network-calls-on-the-client-in-wpf-and-silverlight-and-net-in-general

を使用して、同期コンテキストを設定する必要があります。

<Page.Resources> 
    <ViewModel:ReactiveListViewModel x:Key="model"/> 
</Page.Resources> 

<Grid DataContext="{StaticResource model}"> 
    <Grid.RowDefinitions> 
     <RowDefinition Height="Auto"/> 
     <RowDefinition/> 
    </Grid.RowDefinitions> 
    <Button Content="Start" Command="{Binding StartCommand}"/> 
    <ListBox ItemsSource="{Binding Items}" Grid.Row="1"/> 
</Grid> 

public class ReactiveListViewModel : ViewModelBase 
{ 
    public ReactiveListViewModel() 
    { 
     Items = new ObservableCollection<long>(); 
     StartCommand = new RelayCommand(Start); 
    } 

    public ICommand StartCommand { get; private set; } 

    private void Start() 
    { 
     var observable = Observable.Interval(TimeSpan.FromSeconds(1)); 

     //Exception: This type of CollectionView does not support changes to its SourceCollection from a thread different from the Dispatcher thread. 
     //observable.Subscribe(num => Items.Add(num)); 

     // Works fine 
     observable.ObserveOn(SynchronizationContext.Current).Subscribe(num => Items.Add(num)); 

     // Works fine 
     //observable.ObserveOnDispatcher().Subscribe(num => Items.Add(num)); 
    } 

    public ObservableCollection<long> Items { get; private set; } 
} 
+0

このメソッドはifコードはXamlsコードの背後にありますが、私はエンティティを追加するためにViewModelが必要です。私のソースは、ViewModelから呼び出される非同期サービスです。その時、SynchronitaionContext.CurrentはViewModelの値を持っていません –

+0

私のために働く例を追加しました。 NuGet経由でRx-MainとRx-WPFを追加しました。 – Phil

+0

私のコードは多かれ少なかれ同じですが、私のコレクションのフェッチと割り当てはコンストラクタで起こります。また、クラスを解決するためにUnityを使用しています。私はここに詳細なコードを追加しましたhttp://stackoverflow.com/questions/9377290/synchronizationcontext-current-is-null-on-resolving-with-unity-in-wpf –

1

コードはバックグラウンドスレッドで実行されていますか? UIに影響するため、ViewバインドObservableCollectionはUI /ディスパッチャスレッドでのみ更新できます。

同様の問題については、WPF ObservableCollection Thread Safetyを参照してください。

1

UIへの変更がDispatcherスレッドで実行する必要があります:

は、ここで私の作品例です。ビューモデルを常に変更するanthoerスレッドがある場合は、プロパティーセッターにディスパッチャースレッドを使用させることをお勧めします。この場合、別のスレッドでUI要素を変更しないようにします。

試行:[別のスレッドでのObservableCollectionを更新]の

public string Property 
{ 
    set 
    { 
     Dispatcher.BeginInvoke(()=> _property = value) ; 
     OnPropertyChanged("Property"); 
    } 
    get 
    { 
     return _property; 
    } 
} 
関連する問題