代码:
const a = of([1, 2, 3]);
a.subscribe((data) => console.log('Fairy:' + data));
单步调试,首先执行of所在的index.ts:
data:image/s3,"s3://crabby-images/c012c/c012c38799d57c9ea60bbae60aa1199ff345c390" alt=""
of.js的实现很简单:
data:image/s3,"s3://crabby-images/e6187/e61876855e0291fea45c9abe662e4d79ecd92a8d" alt=""
import { isScheduler } from '../util/isScheduler';
import { fromArray } from './fromArray';
import { scheduleArray } from '../scheduled/scheduleArray';
export function of(...args) {
let scheduler = args[args.length - 1];
if (isScheduler(scheduler)) {
args.pop();
return scheduleArray(args, scheduler);
}
else {
return fromArray(args);
}
}
//# sourceMappingURL=of.js.map
没有scheduler,走fromArray这条路:
data:image/s3,"s3://crabby-images/3dc45/3dc4526f7dc264117cc7023e8e701e7afe9c9f6b" alt=""
fromArray.js的实现:
import { Observable } from '../Observable';
import { subscribeToArray } from '../util/subscribeToArray';
import { scheduleArray } from '../scheduled/scheduleArray';
export function fromArray(input, scheduler) {
if (!scheduler) {
return new Observable(subscribeToArray(input));
}
else {
return scheduleArray(input, scheduler);
}
}
//# sourceMappingURL=fromArray.js.map
data:image/s3,"s3://crabby-images/85c11/85c11ba5c8ff9758e4d98d120fc113e283de260e" alt=""
这个subscribeToArray是一个函数构造器,接收一个数组,返回一个新的函数:
export const subscribeToArray = (array) => (subscriber) => {
for (let i = 0, len = array.length; i < len && !subscriber.closed; i++) {
subscriber.next(array[i]);
}
subscriber.complete();
};
//# sourceMappingURL=subscribeToArray.js.map
data:image/s3,"s3://crabby-images/04577/0457745b24123d94366a772549a5ebe936326ce4" alt=""
该函数和我们在应用代码里调用Observable对象的subscribe方法时传入的回调函数有何区别?
继续往下调试。
data:image/s3,"s3://crabby-images/46af6/46af6a2e582d66826c2fcf55a8d5b5d170cd0aa6" alt=""
调用Observable对象的subscribe方法:
data:image/s3,"s3://crabby-images/0c271/0c271e7f390d9dbdf26971e96b6b81c6fdceb107" alt=""
ObserverOrNext就是上图第63行的箭头函数,error和complete为undefined:
data:image/s3,"s3://crabby-images/5cfd6/5cfd6b5d46511634386d9a8638d2f3eae7095c74" alt=""
新建一个Subscriber对象:
data:image/s3,"s3://crabby-images/8cbe8/8cbe803d983d6d6a025c652857305ee774e00dd4" alt=""
subscriber的父类是subscription:
data:image/s3,"s3://crabby-images/8b69f/8b69f9e13337e1804770f63873bcd532716c0f7b" alt=""
subscriber的destination是一个SafeSubscriber:
data:image/s3,"s3://crabby-images/2ca49/2ca49bac3361aaa37aa2ac58a9b176372ae0845f" alt=""
SafeSubscriber也是一个Subscriber:
data:image/s3,"s3://crabby-images/2b38e/2b38e1b78c4bc5c4dc318a4687fa3344ff349850" alt=""
这个emptyObservable啥实现也没有:
data:image/s3,"s3://crabby-images/d517e/d517ec3b71aa2234055be652b7ba0d7668c37b69" alt=""
import { config } from './config';
import { hostReportError } from './util/hostReportError';
export const empty = {
closed: true,
next(value) { },
error(err) {
if (config.useDeprecatedSynchronousErrorHandling) {
throw err;
}
else {
hostReportError(err);
}
},
complete() { }
};
//# sourceMappingURL=Observer.js.map
ExtensibilityExtensibility
data:image/s3,"s3://crabby-images/55b1b/55b1bb43a872b5f4592ff8740da72491cfbb59c0" alt=""
所以toSubscriber返回的实际是一个subscriber对象:
data:image/s3,"s3://crabby-images/6a9a5/6a9a5b8431eea87ff2a0fe3af312458205069b95" alt=""
data:image/s3,"s3://crabby-images/aec8f/aec8fb8adcf58a62faa39b001633d38a611f5e79" alt=""
首先调用subscriber对象的add方法,目的是通过这个三元表达式,判断到底应该调用subscribe方法,还是trySubscribe方法:
data:image/s3,"s3://crabby-images/ab47a/ab47a5f0dbfa1b94606cfdc45054fa73d4b87854" alt=""
在我的Angular10,执行后者:
data:image/s3,"s3://crabby-images/e1333/e1333665b81c79472e0d4dd0a639fbf5acda4186" alt=""
记住这个语义:Observable的subscribe方法,输入参数为subscriber:
data:image/s3,"s3://crabby-images/bc8f8/bc8f845a20e8f2a3c9ed6041f57831e45f79ff7a" alt=""
_trySubscribe调用_subscribe:
data:image/s3,"s3://crabby-images/b4786/b4786eb0885ff377c7d5b8889369a1559b02432a" alt=""
然后就执行到了之前用subscribeToArray返回的function内部:
data:image/s3,"s3://crabby-images/244f3/244f301f16bda4b0386de67958c592f9672ada40" alt=""
注意在这个上下文里,我们既有输入,又有应用程序传入的subscribe函数,因此可以调用next了:
data:image/s3,"s3://crabby-images/0f5e2/0f5e27835b99005c18c75a56abdd2a1207fe9dcf" alt=""
data:image/s3,"s3://crabby-images/c3996/c3996b9b7354538093d3d6c28f5b5e9ba5ca88dd" alt=""
next和_next的区别就在于有个this.isStopped的判断:
data:image/s3,"s3://crabby-images/0e78d/0e78d72166b21f436de6a7cf0f3007027c344f26" alt=""
注意语义:Observable调用subscribe,而subscriber调用next.
data:image/s3,"s3://crabby-images/48136/4813600a1afaa5c836d949822a29290a510bc6e0" alt=""
subscriber的desination里包含了应用程序传入的callback:
data:image/s3,"s3://crabby-images/4a0f8/4a0f893e6287fce0154e30b81b7ed51791e0f63b" alt=""
data:image/s3,"s3://crabby-images/74a0e/74a0e468c095f40edd5fee04c242dfb0b9b160e6" alt=""
subscriber的_tryOrUnsub函数里,最终调用应用程序的callback:
data:image/s3,"s3://crabby-images/da579/da579f226d63ad5fb82009b072bf91a884d730b0" alt=""
data:image/s3,"s3://crabby-images/88007/8800781392d293ea25fd9a2f98932939b2c65473" alt=""
更多Jerry的原创文章,尽在:"汪子熙":
data:image/s3,"s3://crabby-images/d68b2/d68b247fb4f295cd9ecad4d389742836ad4f9030" alt=""
网友评论