質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
ReactiveX

ReactiveX(Rx、Reactive Extensions)は、リアクティブプログラミングが可能なライブラリ。Java/Android用のRxJava、JavaScript用のRxJSなどさまざまな言語向けに実装されています。

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

Q&A

解決済

1回答

3849閲覧

ObserveCollection<T>/ReadOnlyReactiveCollection<T>から重複を除いたReadOnlyReactiveCollection<string>を作りたい

ry188472

総合スコア74

ReactiveX

ReactiveX(Rx、Reactive Extensions)は、リアクティブプログラミングが可能なライブラリ。Java/Android用のRxJava、JavaScript用のRxJSなどさまざまな言語向けに実装されています。

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

0グッド

0クリップ

投稿2021/10/05 02:52

編集2021/10/05 02:53

Tについて特定の条件を満たす要素は重複とみなして除外したReactiveCollectionを作りたいです。以下のコードでは重複を除外したAddressのコレクションを取得したいと思っています。
いくつか試したのですが、コレクションの変更の通知がうまく行われず0件のままになったりデータが重複したりします。どなたかいい実装方法を教えていただければ幸いです。

c#

1public record Sample(string Id, string Address); 2public class Comp : IEqualityComparer<Sample> 3{ 4 public bool (Sample? x, Sample? y) => x?.Address == y?.Address; 5 public int GetHashCode([DisallowNull] Sample obj) => obj.GetHashCode(); 6} 7 8// 適当なデータ 9// コンストラクタでToReactiveCollectionしたあとにdataの中身が複数件まとめてセットされる 10ObservableCollection<Sample> data; 11 12// ダメな例1 -> Selectの時点でIEnumerable<string>になるためToReadOnlyReactiveCollectionできない 13data.Select(x => x.Address).Distinct().ToReadOnlyReactiveCollection(); 14// ダメな例2 -> Distinctの時点でIEnumerable<Sample>になる 15data.Distinct(new Comp()).ToReadOnlyReactiveCollection(x => x.Address); 16// ダメな例3 -> ToObservableしても変更は通知されない 17data.ToObservable().Select(x => x.Address).Distinct().ToReadOnlyReactiveCollection(); 18// ダメな例4 -> コレクションのAddのたびに走るため件数がおかしくなる 19data.CollectionChangedAsObservable(). 20 SelectMany(_ => data.Select(x => x.Address).Distinct()). 21 ToReadOnlyReactiveCollection();

環境

c# 9.0

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

soi013

2021/10/05 08:11

これは元のObservableCollectionの削除は考慮する必要があるのでしょうか? つまり、以下のようなケースの場合、(1)では、ReadOnlyReactiveCollectionは{a, b}となっていますが、(2)では{b,c}としてほしいのでしょうか?それとも{b}でいいのでしょうか? ``` data.Add(new Sample("a", "12")); data.Add(new Sample("b", "99")); data.Add(new Sample("c", "12")); //(1) data.RemoveAt(0); //(2) ```
ry188472

2021/10/05 08:28

削除は考えていなかったので考慮不要です。
soi013

2021/10/05 08:42

同様に移動(Move)や消去(Reset)、置換(Replace)は考慮しますか? もし純粋に追加(Add)だけでよければ`ObserveAddChanged()`ですむので、シンプルになります
ry188472

2021/10/06 00:40 編集

基本的に毎回洗い替えすることを想定していたので、移動、消去、置換も考慮するつもりはありませんでした。 思ったんですがやはりコレクション操作のそれぞれを購読してフルスキャンしてやるしかないんですかね?ObserveAddChangedだとコレクションに最初にAddするときに100件突っ込むなら100回起きますよね(間違ってたらすみません)。それなら最初にコレクションにデータをつっこむときに重複を除いたObservableCollection<string>を別に作ったほうが効率がいいかなと思ったんですが、どうでしょう。 もしくはObservableCollection<T>でなく、値変更時にPropertyChangedが起きるようにしたT[]のプロパティを作って、要素の増減時は毎回配列を作り直し、そのプロパティからObserveProperty()でReactiveProperty<T[]>や<string>を作ってやるほうがいいのかも・・・
soi013

2021/10/06 00:45

> 毎回洗い替えすることを想定 > コレクション操作のそれぞれを購読してフルスキャンしてやるしかないんですかね? dataに対する全部の操作でReactiveCollectionを作り直す、ということでしょうか?それだったら、`ReadOnlyReactiveProperty<IReadOnlyList<Sample>>`のほうが適切かもしれませんね。 > ObserveAddChangedだとコレクションに最初にAddするときに100件突っ込むなら100回起きますよね はい、ただしそれはObservableColletion自体の特性なので、複数まとめての追加を効率よくパフォーマンスよくやりたかったら、ObservableColletionを使用するのを避けねばなりません。
guest

回答1

0

自己解決

今回のケースでは要素数の変更がほぼ発生しない(変わるときは洗い替え)という条件だったため、INotifyPropertyChangedを実装するModelではObservableCollection<T>をやめてT[]でデータを保持するようにしました。
ただし、Tのメンバの状態監視をする必要がありObserveElementPropertyを使いたかったため、T[]からReadOnlyReactiveCollection<T>の生成を行っています。
具体的なコードは以下です。

c#

1public class Model : INotifyPropertyChanged 2{ 3 public Sample[] Records { get; private set => /* ~~ set時にPropertyChangedが起きるようにする */ } 4}

c#

1private Model _model; 2public ReadOnlyReactiveCollection<Sample> ViewRecords { get; } 3public ReadOnlyReactiveCollection<string> Distincted { get; } 4 5// もととなるデータの配列を監視する 6var ox = _model.ObserveProperty(m => m.Records); // IObservable<Sample[]> 7// 内容を生データと重複除外版とで分ける 8ViewRecords = ox. 9 SelectMany(rec => rec.Select(r => r)). // IObservable<Sample> 10 ToReadOnlyReactiveCollection(); 11Distincted = ox. 12 SelectMany(rec => rec.Select(r => r.Address).Distinct()). // IObservable<string> 13 ToReadOnlyReactivecollection(); 14 15// 変更検知はViewRecordsを監視する 16ViewRecords.ObserveElementProperty(x => x.Id). ~~~

投稿2021/10/06 08:34

編集2021/10/06 08:36
ry188472

総合スコア74

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

soi013

2021/10/06 23:27 編集

質問者様の状況で問題なければ、そのままでよいのですが、いくつか疑問な点があります。 ### 無限に増える ViewRecordsもDistinctedもRecordsが変わるたびに、前のを残したまま追加するので無限に増えますがよいのでしょうか? ### Distinctedが同時追加分だけ? 上のにも関わりますが、重複除く処理`Distinct()`は同時に追加したものだけに適用されていますが、よいのでしょうか? ### 初期値違い `ObserveProperty()`はHotなオペレーターなので、`Records`に元々値が入っていた場合、`ViewRecords`にはそれが反映され、`Distincted`には反映されません。そして記述の順番を逆にすると現象も逆になります。 後の変更でのバグを防ぐためにも、`ox`の変数にとらず、それぞれ`ObserveProperty()`したほうがよいでしょう。
ry188472

2021/10/08 06:31 編集

すみません、確認が遅れました。 > ### 無限に増える これは問題がありました。やっぱりReactiveProperty<T[]>で持たないといけないかも・・・ただそうするとObserveElementPropertyが使えないんですよね。T[]の各要素1つ1つをSubscribeなんかしたくないし、うまい方法があればいいんですが。 > ### Distinctedが同時追加分だけ? Recordsが作り直されたときに、その作り直されたやつに重複の除外はかかるけども、すでにDistinctに追加されてるものと重複しているかどうかまでは判定されないって意味ですよね? > ### 初期値違い 「もともと値が入っていた場合」って初期状態のことを指しますか?それとも2回め以降に値が変わったときのことですか? そもそもHotなオペレーターなら両方に同じ値が流れるんじゃなかったでしたっけ。だからoxに受けていいと思ったんですが
soi013

2021/10/08 14:15 編集

> ObserveElementPropertyが使えない そもそもSampleはrecordなので、中のプロパティは初期化時以降は変更無いのかと思っていました。`ID`は生成後に変更されるのでしょうか?IDが後から変わるという設計は違和感がありますが... > すでにDistinctに追加されてるものと重複しているかどうかまでは判定されないって意味ですよね? はい。 > 「もともと値が入っていた場合」って初期状態のことを指しますか?それとも2回め以降に値が変わったときのことですか? 初期状態ですね。 > そもそもHotなオペレーターなら両方に同じ値が流れるんじゃなかったでしたっけ ややこしいんですが、最初にSubscribeされたときに、現在値を流す、という処理にはその原則は適用されず、本当に"最初"だけになります。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問