【问题标题】:Continue the subscription after error报错后继续订阅
【发布时间】:2017-03-28 00:58:18
【问题描述】:

我有一个代码,我为每个 ID 发出一个 ajax 请求并在收到结果时对其进行处理。这是我的实际代码和jsfiddle 的简单副本:

var ids = [1,2,3,4,5,6];
var ids$ = (id) => {
    return Observable.of(id);
};
var loadIds$ = (id) => {
    if(id == 4) return xhr('/echo/jsoneee/', {id: id});
    return xhr('/echo/json/', {id: id});
};

Observable.from(ids)
.concatMap(id => Observable.forkJoin(ids$(id), loadIds$(id)))
.catch((err, caught) => {
  //get the id for which loadIds failed
  //send the id to subscribe() and continue the subscription
  return Observable.empty();
})
.subscribe(result => console.log(result))

但现在我需要修改代码,以便如果发生错误,我将不得不获取 ajax 请求失败的id,然后继续订阅,就像什么都没发生一样。我还不能做到这一点。非常感谢您的帮助。

【问题讨论】:

  • .concatMap(id => Observable.forkJoin(ids$(id), loadIds$(id)))这应该做什么?
  • @martin 这只是一个让事情变得更简单的模型。但它可以是forkJoinzip 但目的是subscribe() 应该将id 对象和id ajax 请求的响应作为一个项目一起作为一个项目,因为我在订阅和内部进行其他处理我需要这两个数据。

标签: javascript ajax rxjs reactive-programming rxjs5


【解决方案1】:

我认为您可以通过在 Observable.create(...) 中发出正确的值来显着简化此操作:

function xhr(url, json) {
    return Observable.create(function (observer) {
        $.ajax({
            url: url,
            data: JSON.stringify(json),
            success: function (response) {
                observer.next([json.id, json]);
            },
            error: function (jqXHR, status, error) {
                observer.next([json.id]); // <== Notice this
            },
            complete: function () {
                observer.complete();
            }
        });
    });
}

var ids = [1,2,3,4,5,6];
var ids$ = (id) => {
    return Observable.of(id);
};
var loadIds$ = (id) => {
    if(id == 4) return xhr('/echo/jsoneee/', {id: id});
    return xhr('/echo/json/', {id: id});
};

Observable.from(ids)
    .concatMap(id => loadIds$(id))
    .subscribe(result => console.log(result));

这样你就可以完全避免forkJoin()。另请注意,catch() 运算符会自动取消订阅其来源。该运算符旨在继续使用另一个 Observable,因此它现在在您的情况下非常有用。

你当然可以使用:

.catch((error, caught) => {
    return caught;
})

然而,这会导致重新订阅,从而从头开始重新发送所有值,这通常是不希望的。

还有onErrorResumeNext() 运算符可以简单地忽略错误,但这可能不是您想要的。

查看演示:https://jsfiddle.net/4f1zmeyd/1/

一个稍微类似的问题:get new ticket then retry first request

【讨论】:

  • 我宁愿不用根据这个例子修改xhr()。但我不确定为什么这会像我希望的那样与catch 一起工作? jsfiddle.net/ibrahimislam/ecLp6h1u
  • @lbrahim 你不会让它和catch()一起工作,因为它不会做你想做的事。
  • 但是如果你会看到这个例子:jsfiddle.net/ibrahimislam/ecLp6h1u。它按预期工作。
  • @lbrahim 不,不一样。这只是忽略错误并继续,而不在响应中包含失败的 id。只使用onErrorResumeNext() 而不使用catch(),您将获得完全相同的结果
猜你喜欢
  • 1970-01-01
  • 2013-10-18
  • 2013-01-27
  • 2023-04-03
  • 1970-01-01
  • 1970-01-01
  • 2023-04-09
  • 2018-05-23
  • 1970-01-01
相关资源
最近更新 更多