尝试阻止订阅两次时出现 ObjectUnsubscribedError
ObjectUnsubscribedError when trying to prevent subscribing twice
我有一个服务和一个使用它的组件:
PagesService
PagesListComponent
在PagesService
中我有一个Pages
的数组。我通过 BehaviorSubject
通知数组中的更改,他们都订阅了。
PagesService
在 bootstrap
提供,只共享一个实例。那是因为我需要保留数组,而不是每次需要时都下载页面。
代码如下:
pages.service.ts
import { Injectable } from '@angular/core';
import { BehaviorSubject } from 'rxjs/Rx';
import { Http, Response } from '@angular/http';
import { Page } from './../models/page';
@Injectable() export class PagesService {
public pages$: BehaviorSubject<Page[]> = new BehaviorSubject<Page[]>([]);
private pages: Page[] = [];
constructor(private http: Http) { }
getPagesListener() {
return this.pages$;
}
getAll() {
this.http.get('/mockups/pages.json').map((res: Response) => res.json()).subscribe(
res => { this.resetPagesFromJson(res); },
err => { console.log('Pages could not be fetched'); }
);
}
private resetPagesFromJson(pagesArr: Array<any>) {
// Parses de Array<any> and creates an Array<Page>
this.pages$.next(this.pages);
}
}
pages_list.component.ts
import { Component, OnInit } from '@angular/core';
import { Router } from '@angular/router-deprecated';
import { BehaviorSubject } from 'rxjs/Rx';
import { PagesService } from '../../shared/services/pages.service';
import { GoPage } from '../../shared/models/page';
@Component({
moduleId: module.id,
selector: 'go-pages-list',
templateUrl: 'pages_list.component.html',
styleUrls: ['pages_list.component.css']
})
export class PagesListComponent implements OnInit {
pages$: BehaviorSubject<GoPage[]>;
pages: GoPage[];
constructor(private pagesService: PagesService, private router: Router) { }
ngOnInit() {
this.pages$ = this.pagesService.getPagesListener();
this.pages$.subscribe((pages) => { this.pages = pages; console.log(pages) });
this.pagesService.getAll();
}
ngOnDestroy() {
this.pages$.unsubscribe();
}
}
这第一次工作正常,订阅 onInit 和取消订阅 onDestroy。但是当我 return 到列表并尝试再次订阅时(获取 pages[] 的当前值并监听未来的变化),它 引发错误 EXCEPTION: ObjectUnsubscribedError
.
如果我不取消订阅,每次我进入列表时,都会堆叠一个新的订阅,并在收到 next() 时触发所有订阅。
我会通过这种方式获得订阅和取消订阅,而不是直接针对主题:
ngOnInit() {
this.pages$ = this.pagesService.getPagesListener();
this.subscription = this.pages$.subscribe((pages) => { // <-------
this.pages = pages; console.log(pages);
});
this.pagesService.getAll();
}
ngOnDestroy() {
this.subscription.unsubscribe(); // <-------
}
.subscribe() returns 订阅
- 你应该用它来退订
例如
Parent 有一个 reloadSubject: 主题;
- child1 -> 订阅
- child2 -> 订阅
child1 - "WORKS" -> 退订他的订阅
ngOnInit{
sub: Subscription = parent.subscribe();
}
onDestroy{
this.sub.unsubscribe();
}
child2 - "DOES NOT WORK" -> 全部退订 parent
ngOnInit{
parent.subscribe();
}
onDestroy{
parent.unsubscribe();
}
如果您在 parent 上调用取消订阅,两个 children 都消失了。
如果您取消订阅从 .subscribe() 获得的订阅
那么只有一个child退订了。
如有错误请指正!
我有一个服务和一个使用它的组件:
PagesService
PagesListComponent
在PagesService
中我有一个Pages
的数组。我通过 BehaviorSubject
通知数组中的更改,他们都订阅了。
PagesService
在 bootstrap
提供,只共享一个实例。那是因为我需要保留数组,而不是每次需要时都下载页面。
代码如下:
pages.service.ts
import { Injectable } from '@angular/core';
import { BehaviorSubject } from 'rxjs/Rx';
import { Http, Response } from '@angular/http';
import { Page } from './../models/page';
@Injectable() export class PagesService {
public pages$: BehaviorSubject<Page[]> = new BehaviorSubject<Page[]>([]);
private pages: Page[] = [];
constructor(private http: Http) { }
getPagesListener() {
return this.pages$;
}
getAll() {
this.http.get('/mockups/pages.json').map((res: Response) => res.json()).subscribe(
res => { this.resetPagesFromJson(res); },
err => { console.log('Pages could not be fetched'); }
);
}
private resetPagesFromJson(pagesArr: Array<any>) {
// Parses de Array<any> and creates an Array<Page>
this.pages$.next(this.pages);
}
}
pages_list.component.ts
import { Component, OnInit } from '@angular/core';
import { Router } from '@angular/router-deprecated';
import { BehaviorSubject } from 'rxjs/Rx';
import { PagesService } from '../../shared/services/pages.service';
import { GoPage } from '../../shared/models/page';
@Component({
moduleId: module.id,
selector: 'go-pages-list',
templateUrl: 'pages_list.component.html',
styleUrls: ['pages_list.component.css']
})
export class PagesListComponent implements OnInit {
pages$: BehaviorSubject<GoPage[]>;
pages: GoPage[];
constructor(private pagesService: PagesService, private router: Router) { }
ngOnInit() {
this.pages$ = this.pagesService.getPagesListener();
this.pages$.subscribe((pages) => { this.pages = pages; console.log(pages) });
this.pagesService.getAll();
}
ngOnDestroy() {
this.pages$.unsubscribe();
}
}
这第一次工作正常,订阅 onInit 和取消订阅 onDestroy。但是当我 return 到列表并尝试再次订阅时(获取 pages[] 的当前值并监听未来的变化),它 引发错误 EXCEPTION: ObjectUnsubscribedError
.
如果我不取消订阅,每次我进入列表时,都会堆叠一个新的订阅,并在收到 next() 时触发所有订阅。
我会通过这种方式获得订阅和取消订阅,而不是直接针对主题:
ngOnInit() {
this.pages$ = this.pagesService.getPagesListener();
this.subscription = this.pages$.subscribe((pages) => { // <-------
this.pages = pages; console.log(pages);
});
this.pagesService.getAll();
}
ngOnDestroy() {
this.subscription.unsubscribe(); // <-------
}
.subscribe() returns 订阅
- 你应该用它来退订
例如
Parent 有一个 reloadSubject: 主题;
- child1 -> 订阅
- child2 -> 订阅
child1 - "WORKS" -> 退订他的订阅
ngOnInit{
sub: Subscription = parent.subscribe();
}
onDestroy{
this.sub.unsubscribe();
}
child2 - "DOES NOT WORK" -> 全部退订 parent
ngOnInit{
parent.subscribe();
}
onDestroy{
parent.unsubscribe();
}
如果您在 parent 上调用取消订阅,两个 children 都消失了。
如果您取消订阅从 .subscribe() 获得的订阅
那么只有一个child退订了。
如有错误请指正!