Using Observables
- Define the observable/subject/publisher
- Subscribe to the observable
- Emit a new value
- Examples
- Additional Resources
Define the observable/subject/publisher
The terms subject
, publisher
, and observable
are used interchangeably.
A subject
acts as a data source that can emit a sequence of values over time. It serves as a way
to both produce and subscribe to data changes. Subjects are often used in communication between
components, in place of @Input
and @Output
annotations.
A subject
maintains a list of its observers, commonly referred to as subscribers. Whenever the
state changes, the subject
automatically notifies its subscribers.
You can think of subjects like event emitters, that maintain a registry of many listeners. When something happens, the subject emits an event or value to its subscribers.
// score.service.ts
gameStats: GameStats = { score: 0, lines: 0, level: 1, levelUp: 10 };
private scoreSubject = new BehaviorSubject<GameStats>(this.gameStats);
getScoreObservable(): Observable<GameStats> {
return this.scoreSubject.asObservable();
}
In the service, we create a BehaviorSubject
instance, which is a type of subject that requires
an initial value and emits its current value to new subscribers.
Subscribe to the observable
A 'subscription' is an object used to establish a connection between an Observable and its Observer. It allows you to receive values and notifications emitted by the Observable. Once you subscribe to an Observable, the Observer's methods (such as next, error, and complete) get invoked as the Observable emits values or completes.
An Observable instance begins publishing values only when someone subscribes to it. You subscribe by calling the subscribe() method of the instance, passing an observer object to receive the notifications.
// score.component.ts
private scoreService = inject(ScoreService);
gameStats: GameStats = { score: 0, lines: 0, level: 0, levelUp: 0 };
ngOnInit(): void {
this.scoreService.getScoreObservable().subscribe((stats: GameStats) => {
this.gameStats = stats;
})
}
In the component, we subscribe to the observable and update the game stats when a new value is
emitted. The subscribe()
method takes up to three arguments: next
, error
, and complete
. We
only need to handle next
in this example.
Emit a new value
export class ScoreService {
gameStats: GameStats = { score: 0, lines: 0, level: 1, levelUp: 10 };
private scoreSubject = new BehaviorSubject<GameStats>(this.gameStats);
getScoreObservable(): Observable<GameStats> {
return this.scoreSubject.asObservable();
}
updateScore(gameStats: GameStats): void {
this.scoreSubject.next(gameStats);
}
}
Examples
// score.component.ts
export class ScoreService {
gameStats: GameStats = { score: 0, lines: 0, level: 1, levelUp: 10 };
private scoreSubject = new BehaviorSubject<GameStats>(this.gameStats);
getScoreObservable(): Observable<GameStats> {
return this.scoreSubject.asObservable();
}
updateScore(gameStats: GameStats): void {
this.scoreSubject.next(gameStats);
}
}
// score.component.ts
export class ScoreService {
gameStats: GameStats = { score: 0, lines: 0, level: 1, levelUp: 10 };
private scoreSubject = new BehaviorSubject<GameStats>(this.gameStats);
getScoreObservable(): Observable<GameStats> {
return this.scoreSubject.asObservable();
}
}
Basic Example
// Creating an Observable
observable = new Observable(subscriber => {
subscriber.next('Here is a message');
subscriber.next('Here is another');
subscriber.complete();
});
// Subscribing to the Observable
subscription = this.observable.subscribe({
next: value => console.log(value),
complete: () => console.log('Observable completed')
});
Separate Observable Property (Example 1)
gameStats: GameStats = { score: 0, lines: 0, level: 1, levelUp: 10 };
private scoreSubject = new BehaviorSubject<GameStats>(this.gameStats);
getScoreObservable(): Observable<GameStats> {
return this.scoreSubject.asObservable();
}
In this approach, you have a private scoreSubject
property that holds the BehaviorSubject, and
you expose the observable using the getScoreObservable()
method.
Advantages:
- Encapsulation: The observable and its subject are encapsulated within the class. External components can only access the observable through the method.
- Control: You have more control over how the observable is exposed and can apply additional logic if needed before providing access.
Observable as a Property with Direct Access (Example 2)
private _isLoggedIn$ = new BehaviorSubject<boolean>(false);
isLoggedIn$ = this._isLoggedIn$.asObservable();
// access the value with:
const isLoggedIn$ = inject(AuthService).isLoggedIn$;
In this approach, the observable is directly exposed as a public property of the class. There is no need for a separate method to access it.
By exposing it as an observable using asObservable()
, you allow external components or services
to subscribe to changes but they cannot directly modify the BehaviorSubject
. This pattern is
commonly used in Angular services to provide controlled access to internal state while keeping it
private.
Advantages:
- Simplicity: It's a more concise way to provide access to the observable.
- Less Boilerplate: It reduces the amount of code needed to access the observable.
The choice between these two approaches depends on your specific requirements and design preferences. If you want to encapsulate the observable and provide a clear interface for accessing it, the first approach (using a separate method) is more suitable. If simplicity and direct access are sufficient for your use case, the second approach is a valid option. Both approaches are valid, and you can choose the one that best fits your coding style and project needs.