这是我迄今为止开发的代码:
var observable = Observable.Create<string>(async observer =>
{
var wc = new WebClient { UseDefaultCredentials = true };
observer.OnNext(await wc.DownloadStringTaskAsync("http://ya.ru"));
});
observable.Subscribe(
res => Debug.WriteLine("got result: {0}", res),
exc => Debug.WriteLine("exception: {0}", exc.Message)
);
这会正确获取网站的数据并触发我的回调一次。我想要的是有一个像这样的无限循环:等待结果 - &gt;致电OnNext
- &gt;等待n
秒 - &gt;重复一次手术。
在我的Observable.Interval
中创建SelectMany
和Observable
将不会完成,因为这将在一段固定的时间内查询网站。我希望下一次调用仅在上一次成功或失败后触发。实现这一目标的最优雅,最简洁的方法是什么?
答案 0 :(得分:7)
在不改变代码的情况下,你可以在收益后有效地延迟延迟,然后无限期地重复整个事情。
var observable = Observable.Create<string>(async observer =>
{
var wc = new WebClient { UseDefaultCredentials = true };
observer.OnNext(await wc.DownloadStringTaskAsync("http://ya.ru"));
});
observable
.Concat(Observable.Empty<string>().Delay(TimeSpan.FromSeconds(1)))
.Repeat()
.Subscribe(
res => Debug.WriteLine("got result: {0}", res),
exc => Debug.WriteLine("exception: {0}", exc.Message)
);
但是,有更好的方法可以做到这一点,因为在之前的实例中,您每秒都在创建一个新的WebClient。相反,你可以做这样的事情......
using System.Reactive.Threading.Tasks;
var observable = Observable.Using(() => new WebClient(), (client) =>
client.DownloadStringTaskAsync("http://ya.ru")
.ToObservable()
.Concat(Observable.Empty<string>().Delay(TimeSpan.FromSeconds(1)))
.Repeat()
);
如果您想重复错误,可以添加Retry
...
var observable = Observable.Using(() => new WebClient(), (client) =>
client.DownloadStringTaskAsync("http://ya.ru")
.ToObservable()
.Retry(3)
.Concat(Observable.Empty<string>().Delay(TimeSpan.FromSeconds(1)))
.Repeat()
);