Zone.js 简介 & 抛砖引玉
Zone.js是angular团队参照NodeJS的Domain,Dart的Zone,为angular 2开发的核心组件。
一开始,我对Zone.js是拒绝的。我们知道类似的 Domain 模块,主要是为了解决异步错误跟踪问题。所以,当我没有太强烈的错误跟踪需求的时候,Zone.js有啥用?
然而execution context不仅仅可以用来跟踪异步错误,还可以做一些猥琐而实用的事情。
先来理解一下 execution context
Zone.current.fork({}).run(function () {
Zone.current.inTheZone = true;
setTimeout(function () {
console.log('in the zone: ' + !!Zone.current.inTheZone); // 'in the zone: true'
}, 0);
});
console.log('in the zone: ' + !!Zone.current.inTheZone); // 'in the zone: false'
execution context,我们可以理解成只与当前 fork 出来的
Zone实例相关的上下文。
上面的例子很明显,因为只有在 fork 中 Zone 的实例我们才设置了 Zone.current.inTheZone 为 true,所以在外面打印出来的结果是 false。
好神奇,这个怎么做到的呢?
我们想像上面那个过程是同步的,那么发生了什么呢?
const defaultZone = Zone.current
// 生成一个新的Zone
const zone = new Zone()
// 设置当前zone
Zone.current = zone
// 对当前zone设值
Zone.current.inTheZone = true
console.log('in the zone: ' + !!Zone.current.inTheZone)
// 退出当前zone
Zone.current = defaultZone
console.log('in the zone: ' + !!Zone.current.inTheZone)
很好,同步没有什么问题,那么异步怎么办呢?其实很简单,就是在每一个异步入口加一个看门人,就可以了。
const defaultZone = Zone.current
// 生成一个新的Zone
const zone = new Zone()
// 设置当前zone
Zone.current = zone
// 对当前zone设值
Zone.current.inTheZone = true
const anonymousA = function () {
console.log('in the zone: ' + !!Zone.current.inTheZone); // 'in the zone: true'
}
// 给进入异步的函数配发令牌映射到对应zone
anonymousA._zone = zone
// 退出当前zone
Zone.current = defaultZone
setTimeout(() => {
// 函数重新回来,设置当前 zone
Zone.current = anonymousA._zone
anonymousA.call(this)
// 退出当前zone
Zone.current = defaultZone
}, 0)
console.log('in the zone: ' + !!Zone.current.inTheZone)
当然Zone.js实现比上面复杂得多,有兴趣的同学可以看看源代码。
在同一个项目使用不同版本的 jQuery
从上面的例子看,我们可以看到,我们可以在 zone 实例上保存只有该 zone 使用的属性。那么我们在利用Object.defineProperty就可以达成我们的目标了。
- 我们先简单写一个模块执行器(意思是我才不想管加载的事情):
// 写的巨简单,不要吐槽
!function (win, Zone) {
var map = {};
var noop = {};
var dependence = {};
var alias = {};
var hasSet = {};
// 因为懒,仅支持 define(name, factory),反正只是 demo
function define(name, factory) {
if (typeof factory === 'function') {
map[name] = {
factory: factory,
exports: noop
};
} else {
map[name] = {
exports: factory
};
}
}
function require(name) {
var module = map[name]
if (module.exports !== noop) return module.exports;
if (dependence[name]) {
var properties = {};
// 利用Object.defineProperty 组装 window.xxx -> require('xxx') 的映射
Object.keys(dependence[name]).forEach(function (key) {
var res;
if (alias[key]) res = alias[key];
else res = key;
properties[res] = require(key + '@' + dependence[name][key]);
if (!hasSet[res]) {
hasSet[res] = true;
Object.defineProperty(window, res, {
get: function () {
return Zone.current.get(res)
}
});
}
});
// 对每个模块,fork 一个 Zone 实例进行执行
Zone.current.fork({
properties: properties
}).run(function () {
module.exports = module.factory()
});
} else {
module.exports = module.factory();
return module.exports;
}
}
function config(opt) {
Object.assign(dependence, opt.dep);
Object.assign(alias, opt.alias);
}
require.config = config;
window.define = define;
window.require = require;
}(window, Zone)
- 试用一下:
// 模拟两个jQuery
define('jquery@1.4', {
version: '1.4',
bind: function () {
console.log('call bind');
}
})
define('jquery@1.8', {
version: '1.8',
on: function () {
console.log('call on');
}
})
// 仅仅打印版本,不做任何事情
function logVersion() {
console.log('version === ', $.version)
}
// 要运行的第一段代码
define('module1', function module1() {
// 使用1.8版本
$.on();
// 证明即使异步调用,这里面的 $ 依然指向正确
setTimeout(logVersion, 100)
})
// 要运行的第二段代码
define('module2', function module2() {
// 使用1.4版本
$.bind();
// 证明即使异步调用,这里面的 $ 依然指向正确
setTimeout(logVersion, 300)
})
// 载入依赖
require.config({
dep: {
module1: {
'jquery': '1.8'
},
module2: {
'jquery': '1.4'
}
},
alias: {
'jquery': '$'
}
})
require('module1')
require('module2')
具体实现参见:(two-different-jquery)[https://github.com/miniflycn/async-technique-you-may-do-not-know/tree/master/two-different-jquery]
更进一步
其实我们可以基于 Zone.js 做一个 Sandbox,则在大型重历史包袱的应用中,可以很好地将多个技术体系共存而不产生恶心的冲突问题。
或者做一个对任意模块依赖注入的方案,对模块之间做完全解耦。
Zone.js 简介 & 抛砖引玉的更多相关文章
- zone.js - 暴力之美
在ng2的开发过程中,Angular团队为我们带来了一个新的库 – zone.js.zone.js的设计灵感来源于Dart语言,它描述JavaScript执行过程的上下文,可以在异步任务之间进行持久性 ...
- prototype.js简介
prototype.js简介 2007-11-21 14:22 prototype.js是一个很强大的Javascript函数库,它可以让你很轻松的使用一些特效,实现AJAX的功能.虽然prototy ...
- HTML基础--JS简介、基本语法、类型转换、变量、运算符、分支语句、循环语句、数组、函数、函数调用.avi
JS简介 1.JavaScript是个什么东西? 它是个脚本语言,需要有宿主文件,它的宿主文件是HTML文件. 2.它与Java什么关系? 没有什么直接的联系,Java是Sun公司(已被Oracle收 ...
- 01 Node.js简介, 安装&配置
Node.js 简介 Node.js 是什么 Node.js 有着强大而灵活的包管理器(node package manager,npm) 目前, 已经有强大第三方工具模块, 例如数据库连接, 网站开 ...
- Vue.js简介
Vue.js简介 Vue.js的作者为Evan You(尤雨溪),任职于Google Creative Lab,虽然是Vue是一个个人项目,但在发展前景上个人认为绝不输于Google的AngularJ ...
- Zone.js
https://github.com/angular/zone.js/ Zone.js
- Gulp.js简介
Gulp.js简介 我们讨论了很多关于怎么减少页面体积,提高重网站性能的方法.有些是操作是一劳永逸的,如开启服务器的gzip压缩,使用适当的图片格式,或删除一些不必要的字符.但有一些任务是每次工作都必 ...
- angular 2+ 变化检测系列三(Zone.js在Angular中的应用)
在系列一中,我们提到Zone.js,Zones是一种执行上下文,它允许我们设置钩子函数在我们的异步任务的开始位置和结束位置,Angular正是利用了这一特性从而实现了变更检测. Zones.js非常适 ...
- 《React Native 精解与实战》书籍连载「Node.js 简介与 React Native 开发环境配置」
此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...
随机推荐
- 给AOP的after函数使用原函数局部变量
引:AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术. 这里我们可以理解为在执行某函数时,要先执 ...
- [转]Oracle数据泵的使用
使用数据泵导出数据 1.连接Oracle数据库SQL> conn / as sysdba已连接. 2.创建一个操作目录SQL> create directory dump_dir as ' ...
- 『TCP/IP详解——卷一:协议』读书笔记——03
2013-08-17 17:31:49 1.7 分用 分用(Demultiplexing):这是一个过程——当目的主机收到一个以太网数据帧时,数据就开始从协议栈中由底向上升,同时去掉各层协议上的报文首 ...
- 命令行导入SQL文件
摘要:把数据库导出为XX.sql格式的数据库文件,导入到另外一个数据库中的时候,总是无法全部导入.及时用mysql的命令界面导入依然是无法全部导入.老师告诉我:在命令行中运行的效率和成功率是最快和最高 ...
- 快速创建node应用[Express框架]
安装Express npm install -g express 建立工程 express -e ejs FaceExpresscd FaceExpress && npm instal ...
- CPU相关知识---物理CPU数、物理核数、逻辑核数、逻辑CPU数 ?
一.物理CPU数.物理核数.逻辑核数.逻辑CPU数 相互关系??? 物理CPU数 ---> 每个物理CPU对应物理核数 ---> (每个物理核数对应逻辑核数)物理CPU对应逻辑核数 --- ...
- iOS自动适配
自iphone4s以后,苹果先后推出了iphone5.iphone5s.iphone6.iphone6plus.iphone6s.iphone6splus这些新的机型,它们的屏幕大小各有所异,从此给我 ...
- 尝试在Linux上编译KestrelHttpServer
Kestrel是目前在非Windows平台上运行ASP.NET 5应用程序的唯一可用Web服务器,但微软似乎将它冷落在一边,源代码更新很慢. 今天试着在Linux上编译Kestrel的源代码,遇到了很 ...
- 真实世界:使用WCF扩展在方法调用前初始化环境
OperationInvoker 介绍 OperationInvoker 是 WCF 运行时模型中在调用最终用户代码前的最后一个扩展点,OperationInvoker 负责最终调用 Service ...
- 新手容易混乱的String+和StringBuffer,以及Java的方法参数传递方式。
之前在交流群里和猿友们讨论string+和stringbuffer哪个速度快以及Java的方法参数传递的问题,引起了群里猿友的小讨论.最终LZ得出的结果是string+没有stringbuffer快, ...