javascript语言本身没有实现模块(包)机制,而各个js框架都有自己的实现,简单点的如extjs,yui2,直接利用对象属性一个namespace函数搞定,其中我觉得最复杂最精巧的要算yui3的动态加载模块机制,其与python的内部模块加载实现机制有异曲同工之妙,足以证明javascript也可以实现复杂的语言基础架构。
yui3 模块机制详见:YUI3 use&Loader 简析
0。模块与包没有明显的界限
python
包(package)也是模块(module),包也要由__init__.py来标识,只不过包
1。提供了模块的约束命名空间
2。提供了模块的搜索加载路径
javascript
语言没有包的概念,框架实现的包也只不过用来约束其下面的模块命名空间。
1.模块缓存与集中管理
python
所有已加载模块位于 sys.modules,为dict结构,对应
{模块名fullname:模块对象<module>}
import 已加载模块只不过相当于将 sys.module中对应模块引到local命名空间,多个文件多次import同一模块内存中为同一模块对象。
yui3
所有已加载模块位于
YUI.Enc.Mods =[{ name: name, fn: fn, //模块初始化的代码 version: version, details: details || {} } ....]
但是yui3 import(use)时需要 attach操作,即加载与初始化分离,运行 fn 于当前模拟命名空间(YUI()实例),在attach中才实际进行模块绑定操作,部分原因也是由于yui3模块的编写机制,即attach实际上运行模块的初始化函数
Y.moduleName=moduleFunction
则造成虽然可以一次加载,但是需要多次初始化问题:每次import(use),实际上模块不同,内存中存在同一模块的多个function copy。
2.模块路径搜索
python
通过默认系统搜索路径 sys.path 以及包搜索路径 __path__,来实现透明加载模块名的相应功能函数。
yui3
通过种子脚本路径确认系统模块搜索路径以及YUI()构造函数中指定模块的配置路径,通过动态添加 script 节点来下载脚本文件,进而加载模块,虽然需要手工指定自有模块路径,但是use时可透明实现对应模块名导入。
3.模块使用
python
语言级别的名字空间实现,可约束模块于import所在文件的对应作用域。
yui3
所有脚本文件共享统一命名空间,通过对象object来限制模块的作用范围,即只能通过 YUI()示例来访问模块,而YUI()实例则通过函数来减少其对全局命名空间的暴露,但是很大程度上依赖于人为规范约束。