我被rx和特定查詢卡住了。 問題:Rx groupby直到條件發生變化
許多單一的更新操作是由連續流產生的。操作可以是插入或刪除。我想緩衝這些數據流並在當時執行少量操作,但維護訂單非常重要。此外,操作應被緩衝並在序列來完成每X秒
實施例:
在:
insert-insert-insert-delete-delete-insert-delete-delete-delete-delete
輸出:
insert(3)-delete(2)-insert(1)-delete(4)
我寫了一個簡單的應用程序,以測試它,它的作品或多或少,因爲我會,但它不尊重傳入的插入/刪除順序
namespace RxTests
{
using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Reactive.Concurrency;
using System.Reactive.Linq;
using System.Reactive.Subjects;
using System.Text;
using System.Threading;
internal class Program
{
private static readonly Random Random = new Random();
private static readonly CancellationTokenSource ProducerStopped = new CancellationTokenSource();
private static readonly ISubject<UpdateOperation> operations = new Subject<UpdateOperation>();
private static void Main(string[] args)
{
Console.WriteLine("Starting production");
var producerScheduler = new EventLoopScheduler();
var consumerScheduler = new EventLoopScheduler();
var producer =
Observable.Interval(TimeSpan.FromSeconds(2))
.SubscribeOn(producerScheduler)
.Subscribe(Produce, WriteProductionCompleted);
var consumer =
operations.ObserveOn(producerScheduler)
.GroupBy(operation => operation.Delete)
.SelectMany(observable => observable.Buffer(TimeSpan.FromSeconds(8), 50))
.SubscribeOn(consumerScheduler)
.Subscribe(WriteUpdateOperations);
Console.WriteLine("Type any key to stop");
Console.ReadKey();
consumer.Dispose();
producer.Dispose();
}
private static void Produce(long time)
{
var delete = Random.NextDouble() < 0.5;
Console.WriteLine("Produce {0}, {1} at {2}", time + 1, delete, time);
var idString = (time + 1).ToString(CultureInfo.InvariantCulture);
var id = time + 1;
operations.OnNext(
new UpdateOperation(id, delete, idString, time.ToString(CultureInfo.InvariantCulture)));
}
private static void WriteProductionCompleted()
{
Console.WriteLine("Production completed");
ProducerStopped.Cancel();
}
private static void WriteUpdateOperation(UpdateOperation updateOperation)
{
Console.WriteLine("Consuming {0}", updateOperation);
}
private static void WriteUpdateOperations(IList<UpdateOperation> updateOperation)
{
foreach (var operation in updateOperation)
{
WriteUpdateOperation(operation);
}
}
private class UpdateOperation
{
public UpdateOperation(long id, bool delete, params string[] changes)
{
this.Id = id;
this.Delete = delete;
this.Changes = new List<string>(changes ?? Enumerable.Empty<string>());
}
public bool Delete { get; set; }
public long Id { get; private set; }
public IList<string> Changes { get; private set; }
public override string ToString()
{
var stringBuilder = new StringBuilder("{UpdateOperation ");
stringBuilder.AppendFormat("Id: {0}, Delete: {1}, Changes: [", this.Id, this.Delete);
if (this.Changes.Count > 0)
{
stringBuilder.Append(this.Changes.First());
foreach (var change in this.Changes.Skip(1))
{
stringBuilder.AppendFormat(", {0}", change);
}
}
stringBuilder.Append("]}");
return stringBuilder.ToString();
}
}
}
}
任何人可以幫助我正確的查詢?
感謝
UPDATE 13年3月8日(由JerKimball建議)
以下線的微小變化/補充JerKimball代碼打印結果:
using(query.Subscribe(Print))
{
Console.ReadLine();
producer.Dispose();
}
使用以下打印方法:
private static void Print(IObservable<IList<Operation>> operations)
{
operations.Subscribe(Print);
}
private static void Print(IList<Operation> operations)
{
var stringBuilder = new StringBuilder("[");
if (operations.Count > 0)
{
stringBuilder.Append(operations.First());
foreach (var item in operations.Skip(1))
{
stringBuilder.AppendFormat(", {0}", item);
}
}
stringBuilder.Append("]");
Console.WriteLine(stringBuilder);
}
和下面的字符串的操作:
public override string ToString()
{
return string.Format("{0}:{1}", this.Type, this.Seq);
}
順序被保存,但是:
- 我不知道其他內部認購訂閱:這是正確的(這是一個問題,因爲我有很久以前,對我而言並不清楚)?
- 我總是有每個列表(即使流產生兩個以上的連續值與同類型)的不超過兩種元素
'我不知道在另一個訂閱訂閱:它是正確的< - 你是什麼意思?這發生在哪裏? (編輯)哦,我看到 - 在你的'打印'...是的,你不想這樣做,如果只是因爲你現在正在泄漏一個'IDisposable'' – JerKimball 2013-03-08 20:14:20
'Subscribe'有一個重載需要一個'Action' - 使用它;這就是爲什麼我在我的例子中使用'Console.WriteLine'。如果你改變'Print'的簽名來取'IList ',你可以'使用(query.Subscribe(Print))' –
JerKimball
2013-03-08 20:17:12
從Select我得到一個IObservable >>,這就是爲什麼我用第一次打印(IObservable >操作)。所以我會再次包裝第二個操作。訂閱(打印)在使用 –
fra
2013-03-08 20:22:01