我正在尝试并行执行大量网络操作,并且我想为每个操作设置超时。
由于 Parallel.ForEach 没有简单的超时选项,所以我使用的是 System.Reactive。
这是我的代码:
public void networkOps(List<MacCpe> source, Action<List<Router>, List<Exception>> onDone) {
var routers = new List<Router>();
var exceptions = new List<Exception>();
Observable.Defer(() => source.ToObservable())
.ObserveOn(Scheduler.CurrentThread)
.SubscribeOn(Scheduler.Default)
.SelectMany(it =>
Observable.Amb(
Observable.Start(() => {
switch(it.type) {
case AntennaType.type1: {
//network stuff
}
break;
case AntennaType.type2: {
//network stuff
}
break;
case AntennaType.type3: {
//network stuff
}
break;
case AntennaType.type4: {
//network stuff
}
break;
default: throw new NullReferenceException("Nothing");
}
}).Select(_ => true),
Observable.Timer(TimeSpan.FromSeconds(60)).Select(_ => false)
),
(it, result) => new { it, result }
)
.Subscribe (
x => {
Console.WriteLine("checked item number " + x.it.Id);
},
ex => {
Console.WriteLine("error string");
}, () => {
onDone(routers, exceptions);
}
);
}
我正在使用 Observable.Amb 运算符并行运行 60 秒计时器,该计时器用作超时。
但是,当我运行此方法时,程序会立即退出,而不会进入 onDone 的回调。
我在网上看到,我可以使用 ObserveOnDispatcher 在 Ui 线程上观察,同时在线程池上运行阻塞代码,但我在终端应用程序服务器端的 Linux 上的 dotnet core 上使用它。
如何在控制台应用程序中的"主线程"上进行观察?
提前感谢您的回复。
当您替换Parallel.ForEach
听起来您很高兴进行阻塞操作时。按照您设置的方式使用 Rx,它不是阻塞操作,因此该方法会立即结束。
修复起来非常简单。只需将您的.Subscribe
更改为以下内容:
.Do(
x =>
{
Console.WriteLine("checked item number " + x.it.Id);
},
ex =>
{
Console.WriteLine("error string");
}, () =>
{
onDone(routers, exceptions);
}
)
.Wait();
我也会摆脱你的.ObserveOn(Scheduler.CurrentThread)
和.SubscribeOn(Scheduler.Default)
,直到你确定你需要这些。