【发布时间】:2020-08-23 10:20:21
【问题描述】:
我有一个服务负责使用计时器每隔 x 秒执行一次 httpClient.get。 我需要这个计时器在服务启动时开始运行,所以 timer 是在 service constructor 中定义的。根据我的理解,订阅应该注册在timer范围内,如下图(没必要我不想改,除非写错了)。
只要后端服务器没有错误\异常\错误500异常,所有系统都可以正常工作。 现在,我需要两件事:
- 每当后端服务器出现问题时,我想catchError。
- 我希望观察者根据计时器时间继续运行(到下一个滴答声),即使有异常。 每当出现异常时,我的最终结果应该是到达组件中的 popUpAlert 查看我的代码 - 这是 webapi 控制器:
public IActionResult getSomeErrorAsTest()
{
try
{
throw new Exception("Serer error");
}
catch(Exception ex)
{
return StatusCode(StatusCodes.Status500InternalServerError, new List<string>());
//throw ex;
}
}
这就是服务(假设每个get请求中的数据都会发生变化——如果真的发生了就不需要实现):
export class MyService
{
MyDataSubject = new Subject<any[]>();
MyDataChanged :Observable>any[]> = this.MyDataSubject.asObservable();
subscribe :Subscription;
constructor(private httpClient : HttpClient)
{
this.subscribe = timer(0, 30000).pipe(
switchMap(()=>
this.getData())).subscribe();
}
getData()
{
return this.httpClient.get<any[]>(<controller url>)
.pipe(
tap(res =>
{
this.MyDataSubject.next(res);
}),
catchError(error =>
{
debugger;//I would expect to catch the debugger here, but nothing happens
return throwError(error);
})
)
}
}
消费者组件:
export class MyComponent (private mySrv : MyService)
{
getMyData()
{
let sub =this.mySrv.MyDataChanged.subscribe(result => doSomething(),
error=> popUpAlert());
}
}
【问题讨论】:
-
我会将整个错误处理转移到服务中。在您的服务中调用
popUpAlert()incatchError并返回of(null)。当结果为null但没有错误情况时,让您的组件处理这种情况。在这里查看我的答案:stackoverflow.com/a/54790621/9423231 -
1)
MyDataSubject不知道抛出了任何错误。因为它在另一个流中抛出 2)您无法捕获throwError运算符抛出的错误。它不会导致错误,而是会发出一个值。 -
与您的问题无关(Rafi 的回答一针见血),但 Angular 中的服务支持一些与组件相同的生命周期回调(包括 OnInit 和 OnDestroy)。您最好在 init 挂钩中启动计时器并在 destory 挂钩中销毁订阅。
标签: angular timer rxjs rxjs-observables rxjs-pipe