در طراحی برنامههای Angular توصیه شدهاست تا هرگونه منطقی که مستقیما به View یک کامپوننت مرتبط نیست، به یک کلاس سرویس منتقل شود. در این بین ممکن است نیاز به صدور رخدادی از یک سرویس به خارج از آن باشد؛ چیزی مانند EventEmitter. اما EventEmitter برای سرویسها
طراحی نشدهاست و کاربرد صحیح آن صرفا محدود به کامپوننتها است. برای حل این مساله، API سرویس ما باید یک Observable را در معرض دید استفاده کننده قرار دهد تا توسط آن بتوان رخدادهایی را به کامپوننتهای مشترک شدهی به آن، صادر کرد.
چگونه میتوان رخدادهایی از نوع Observable را ایجاد کرد؟
کلاس Subject پاسخی است به این پرسش. Subjectها Observableهایی هستند که میتوانند چندین مشترک داشته باشند و رخدادهایی را به مشترکین خود صادر کنند. برای کار با آنها باید یک private Subject را در سرویس خود ایجاد کرد و سپس جریان منتقل شدهی توسط آنرا توسط یک public Observable در اختیار مصرف کنندگان قرار داد. با فراخوانی متد next یک Subject، رخدادی به مشترکین آن منتقل میشود.
import { Subject } from “rxjs/Subject”;
public countdown: number = 0;
private countdownEndSource = new Subject<void>();
public countdownEnd$ = this.countdownEndSource.asObservable();
مرسوم است نام Observableهایی را که قرار است رخدادی را صادر کنند به $ ختم میکنند.
استفاده کنندگان نیز مشترک این $countdownEnd شده و هر بار که در طرف سرویس، متد next آن فراخوانی میشود، از به روز رسانی آن مطلع خواهند شد.
چرا مستقیما از مقدار countdown استفاده نکنیم؟
در قسمتی از سرویس فوق که ملاحظه میکنید، میتوان مقدار countdown را مستقیما نیز در کامپوننتها مورد استفاده قرار داد. اما این روش بهینه نیست. از این جهت که Angular باید مدام تغییرات این خاصیت را رصد کند و به آن واکنش نشان دهد. آیا بهتر نیست ما به Angular اعلام کنیم که مقدار آن تغییر کردهاست و اکنون بهتر است View را به روز رسانی کنی؟ با ارائهی مقادیر جدیدی توسط یک Observable، اکنون Angular صرفا به تغییرات آن واکنش نشان خواهد داد و دیگری نیاز به بررسی مداوم تغییرات مقدار countdown ندارد.
یک مشکل! Subject تعریف شده، مقادیر را تنها در زمان فراخوانی متد next ارائه میدهد و نه به صورت دیگری.
پیشتر با دسترسی مستقیم به خاصیت countdown، همواره به مقادیر آن هم دسترسی داشتیم. اما با استفاده از یک Subject، تنها زمانیکه متد next آن فراخوانی شود میتوان به این مقدار دسترسی یافت. برای رفع این مشکل یک Subject ویژه به نام BehaviorSubject طراحی شدهاست که به محض مشترک شدن به آن، اولین و یا آخرین مقدار آنرا میتوان دریافت کرد.
تفاوت Subject با BehaviorSubject
BehaviorSubject مانند یک Subject است؛ با این تفاوت که همواره از وضعیت خود آگاه میباشد. یک BehaviorSubject:
- همواره دارای مقداری است. حتی در زمان وهله سازی، باید مقدار اولیهای را برای آن مشخص کرد.
- در زمان اشتراک به آن، میتوان آخرین مقدار موجود در آن را که ممکن است اولین مقدار آن نیز باشد، دریافت کرد.
- همواره میتوان مقدار آنرا توسط متد getValue بدست آورد.
و مهمترین مزیت آن نسبت به Subject، همان مورد دوم است. اگر مشترک یک Subject شویم، تا متد next آن فراخوانی نشود، مقداری را دریافت نمیکنیم. اما همان لحظه که مشترک BehaviorSubject میشویم، آخرین مقدار موجود در آنرا دریافت خواهیم کرد.
برای مثال فرض کنید کامپوننتی را دارید که به خاصیت isLoggedIn از نوع Observable یک Subject گوش فرا میدهد. اما اشتراک آن پس از فراخوانی متد next در این سرویس بودهاست. از این رو این کامپوننت هیچگاه متوجه تغییر و یا مقدار نهایی isLoggedIn نخواهد شد. به همین جهت است که به BehaviorSubject نیاز داریم. در این بین مهم نیست که چه زمانی مشترک آن میشویم؛ همواره در زمان اشتراک، آخرین و یا اولین مقدار موجود در آنرا دریافت خواهیم کرد.
یک مثال: بررسی عملکرد BehaviorSubject
در ادامه یک ماژول را به همراه 4 کامپوننت و یک سرویس سفارشی ایجاد میکنیم:
ng g m ServiceComponentCommunication -m app.module --routing
ng g c ServiceComponentCommunication/First
ng g c ServiceComponentCommunication/Second
ng g c ServiceComponentCommunication/Third
ng g c ServiceComponentCommunication/Final
ng g s ServiceComponentCommunication/Sample
هدف این است که سه کامپوننت اول، دوم و سوم را در کامپوننت final، همانند تصویر فوق نمایش دهیم.
در این بین یک سرویس انتشار اطلاعات نیز طراحی شدهاست:
import { Injectable } from "@angular/core";
import { BehaviorSubject } from "rxjs/BehaviorSubject";
@Injectable()
export class SampleService {
private msgSource = new BehaviorSubject<string>("default service value");
telecast$ = this.msgSource.asObservable();
constructor() { }
editMsg(newMsg: string) {
this.msgSource.next(newMsg);
}
}
کار این سرویس ارائه یک پیام از نوع BehaviorSubject از طریق خاصیت عمومی $telecast آن است که به صورت Observable در معرض دید کامپوننتهای مشترک به آن قرار خواهد گرفت. هدف این است که کامپوننتها مدام تغییرات msg را بررسی نکنند و فقط به آخرین تغییر صادر شدهی توسط کامپوننت که از طریق فراخوانی متد next در متد editMsg صورت میگیرد، واکنش نشان دهند.
در کامپوننت اول، نحوهی اشتراک به این سرویس را مشاهده میکنید:
import { SampleService } from "./../sample.service";
import { Component, OnInit, OnDestroy } from "@angular/core";
import { Subscription } from "rxjs/Subscription";
@Component({
selector: "app-first",
templateUrl: "./first.component.html",
styleUrls: ["./first.component.css"]
})
export class FirstComponent implements OnInit, OnDestroy {
editedMsg: string;
sampleSubscription: Subscription;
constructor(private sampleService: SampleService) { }
ngOnInit() {
this.sampleSubscription = this.sampleService.telecast$.subscribe(message => {
this.editedMsg = message;
});
}
editMsg() {
this.sampleService.editMsg(this.editedMsg);
}
ngOnDestroy() {
this.sampleSubscription.unsubscribe();
}
}
کار اشتراک در این کامپوننت در متد ngOnInit انجام شدهاست. بسیار مهم است جهت عدم بروز نشتی حافظه، در متد ngOnDestroy کار unsubscribe بر روی این اشتراک نیز صورت گیرد.
در اینجا هر زمانیکه متد next در سرویس فراخوانی شود، this.editedMsg مقدار جدیدی را دریافت میکند.
با این قالب:
<div class="panel panel-default">
<div class="panel-heading">
<h2 class="panel-title">First Component</h2>
</div>
<div class="panel-body">
<p> {{editedMsg}}</p>
<input class="form-control" type="text" [(ngModel)]="editedMsg">
<button (click)="editMsg()" class="btn btn-primary">Change</button>
</div>
</div>
اما اگر به تصویر دقت کنید، this.editedMsg هم اکنون دارای مقدار است (در اولین بار اجرای این کامپوننت). علت آن به داشتن مقدار اولیهای در BehaviorSubject تعریف شده بر میگردد که در اولین بار اشتراک به آن، در اختیار مشترک قرار خواهد گرفت. این مورد، مهمترین تفاوت BehaviorSubject با Subject است.
در این کامپوننت اگر کاربر مقداری را در textbox وارد کند و سپس بر روی دکمهی Change کلیک نماید، این تغییر از طریق سرویس، به تمام مشترکین آن صادر خواهد شد.
کامپوننت دوم نیز مانند کامپوننت اول است، فقط یک textbox ورود اطلاعات را به همراه ندارد.
همانطور که ملاحظه میکنید، این کامپوننت نیز دارای مقدار اولیهی BehaviorSubject است.
کامپوننت سوم، اندکی متفاوت است:
import { SampleService } from "./../sample.service";
import { Component, OnInit, OnDestroy } from "@angular/core";
import { Subscription } from "rxjs/Subscription";
@Component({
selector: "app-third",
templateUrl: "./third.component.html",
styleUrls: ["./third.component.css"]
})
export class ThirdComponent implements OnInit, OnDestroy {
message: string;
sampleSubscription: Subscription;
constructor(private sampleService: SampleService) { }
ngOnInit() {
}
subscribe() {
this.sampleSubscription = this.sampleService.telecast$.subscribe(message => {
this.message = message;
});
}
ngOnDestroy() {
if (this.sampleSubscription) {
this.sampleSubscription.unsubscribe();
}
}
}
در اینجا کار اشتراک در متد subscribe فراخوانی شدهی توسط قالب آن صورت میگیرد:
<div class="panel panel-default">
<div class="panel-heading">
<h2 class="panel-title">Third Component</h2>
</div>
<div class="panel-body">
<p>{{message}}</p>
<button (click)="subscribe()" class="btn btn-success">Subscribe</button>
</div>
</div>
و چون این متد پس از ngOnInit قرار است توسط کاربر فراخوانی شود، مقدار message این کامپوننت هنوز خالی است.
اکنون اگر بر روی دکمهی Subscribe آن کلیک کنیم، بلافاصله در لحظهی اشتراک، اولین/آخرین مقدار موجود در BehaviorSubject را دریافت خواهیم کرد:
کامپوننت Final نیز تمام کامپوننتها را در صفحه نمایش میدهد:
<div class="row">
<div class="col-md-4">
<app-first></app-first>
</div>
<div class="col-md-4">
<app-second></app-second>
</div>
<div class="col-md-4">
<app-third></app-third>
</div>
</div>
و اگر در textbox کامپوننت اول، مقدار Test را وارد کنیم و سپس بر روی دکمهی Change آن کلیک نمائیم، این مقدار به تمام کامپوننتهای مشترک به BehaviorSubject سرویس برنامه، منتشر خواهد شد:
کدهای کامل این مثال را از اینجا میتوانید دریافت کنید.