javascript模块化编程的cmd规范(sea.js)
CMD(Common Module Definition,通用模块定义)是一种模块定义规范,规范中明确了模块的基本书写格式和基本交互规则。SeaJS就是遵循的这个规范。
define函数
在CMD规范中,一个模块就是一个文件,模块的区分通过define关键字来定义,最基本的格式是:
define(factory); // 这里的define是一个全局函数,factory可以是函数、对象或字符串
define接受factory参数,factory可以是一个函数,也可以是一个对象或字符串。
当factory为对象、字符串时,表示模块的接口就是该对象、字符串。
比如可以如下定义一个JSON数据模块:
define({ "foo": "bar" });
也可以通过字符串定义模板模块:
define('I am a template. My name is {{name}}.');
当factory为函数时,表示是模块的构造方法。执行该构造方法,可以得到模块向外提供的接口。
当factory方法在执行时,默认会传入三个参数:require、exports和module:
define(function(require, exports, module) {
// 模块代码
});
define也可以接受两个以上的参数,进阶的格式是:
define(id, deps, factory); // id是模块标识字符串,deps是模块依赖数组
参数中,字符串id表示模块标注,数组deps表示模块依赖。
define('helloworld', ['jquery'], function(require, exports, module) {
// 模块代码
});
id和deps参数是可以省略的,省略时可以通过构建工具自动生成。注意的是带id和deps参数的define用法不属于CMD规范,而属于Modules/Transport规范。
define函数带有成员属性cmd,是一个空对象,可用来判断当前页面是否有CMD模块加载器。
if (typeof define === "function" && define.cmd) {
// 有 Sea.js 等 CMD 模块加载器存在时的逻辑代码
}
require函数(factory函数的参数之一)
require是factory函数的第一个参数,它有下列的用法:
require(id),通过接受模块标识作为唯一参数,用来获取其他模块提供的接口。
define(function(require, exports) {
// 获取模块 a 的接口
var a = require('./helloworlda'); // 调用模块 helloworld 的方法
a.doSomething();
});
require.async(id, callback),用来在模块内部加载模块,并在加载完成后执行指定回调。callback参数可选。
define(function(require, exports, module) {
// 异步加载一个模块,在加载完成时,执行回调
require.async('./b', function(b) {
b.doSomething();
}); // 异步加载多个模块,在加载完成时,执行回调
require.async(['./c', './d'], function(c, d) {
c.doSomething();
d.doSomething();
});
});
与上一个用法不同的是,前者是同步往下执行,后者是异步回调执行,一般用来加载可延迟异步加载的模块。
require.resolve(id),使用模块系统内部的路径解析机制来解析并返回模块路径,该函数并不会加载模块,只返回解析后的绝对路径。
define(function(require, exports) {
console.log(require.resolve('./b'));// http://path/b.js
});
这样就可以获取模块的路径,一般用在插件环境或需要动态拼接模块路径的场景下。
exports对象(factory函数的参数之一)
exports是一个对象,用来向外提供模块接口。
define(function(require, exports) {
// 对外提供 foo 属性
exports.foo = 'bar'; // 对外提供 doSomething 方法
exports.doSomething = function() {};
});
除了给exports对象增加成员,还可以使用return直接向外提供接口。
define(function(require) {
// 通过 return 直接提供接口
return {
foo: 'bar',
doSomething: function() {}
};
});
如果return语句是模块中的唯一代码,还可以简化为(factory为对象或字符串):
define({
foo: 'bar',
doSomething: function() {}
});
上面这种格式特别适合定义一个JSONP模块。
要注意的是,exports仅仅是module.exports的一个引用,在factory内部给exports重新赋值时,并不会改变module.exports的值,因此给exports赋值是无效的,不能用来更改模块接口。比如以下的写法就是错误的:
define(function(require, exports) {
// 错误用法!!!
exports = { // 应该改为module.exports = {}
foo: 'bar',
doSomething: function() {}
};
});
module对象(factory函数的参数之一)
module是一个对象,存储与当前模块相关的一些属性和方法。
module.id,模块的唯一标识。
define('yanggb', [], function(require, exports, module) {
// 模块代码
console.log(module.id); // yanggb
});
define函数的第一个参数就是模块唯一标识,用module取得时候就能得到yanggb。
module.uri,模块的绝对路径,根据模块系统的路径解析规则得到。
define(function(require, exports, module) {
// 模块代码
console.log(module.uri); // http://path/file.js
});
一般情况下(没有在define函数中自定义id参数),module.id的值就是module.uri的值。
module.dependencies,模块的依赖,是一个数组。
module.exports,模块对外提供的接口。
传给factory函数的exports参数是module.exports对象的一个引用。只通过exports参数来提供接口,有时无法满足开发者的所有需求。例如当模块的接口是某个类的实例时,就需要通过module.exports来实现:
define(function(require, exports, module) {
// exports 是 module.exports 的一个引用
console.log(module.exports === exports); // true // 重新给 module.exports 赋值
module.exports = new SomeClass(); // exports 不再等于 module.exports
console.log(module.exports === exports); // false
});
要注意的是,对module.exports的赋值必须要同步执行,不能放到回调函数里(异步)。
// x.js
define(function(require, exports, module) {
// 错误用法
setTimeout(function() {
module.exports = {a: "hello"};
}, 0);
}); // y.js,在 y.js 里有调用到上面的 x.js:
define(function(require, exports, module) {
var x = require('./x'); // 无法立刻得到模块 x 的属性 a (有延迟)
console.log(x.a); // undefined
});
总结
这里学习的顺序就是:define函数 -> factory函数(define函数的参数) -> require函数、exports对象和modules对象(factory函数的参数)。其实用法也比较简单,对开发很有帮助,有助于开发思路的整理和后期代码的维护。优点也就是模块化编程的优点。
"我唱的不够动人你别皱眉。"
javascript模块化编程的cmd规范(sea.js)的更多相关文章
- Javascript模块化编程(二)AMD规范(规范使用模块)
这个系列的第一部分介绍了Javascript模块的基本写法,今天介绍如何规范地使用模块,先想一想,为什么模块很重要?接下来为您详细介绍,感兴趣的朋友可以了解下啊.今天介绍如何规范地使用模块. 七.模块 ...
- Javascript模块化编程:AMD规范及require.js用法【转】 - loheonly的笔记 - 前端网(W3Cfuns)
http://www.w3cfuns.com/blog-5425789-5399326.html
- 坑人的 Javascript 模块化编程 sea.js
坑人的 Javascript 模块化编程 sea.js 忧伤 加 蛋疼的 开始了 看文档 Sea.js 进行配置 seajs.config({ // 设置路径,方便跨目录调用 paths: { 'ar ...
- javascript模块化编程思想、实现与规范
随着BS架构的发展,网站逐渐变成了互联网应用程序,嵌入网络的JavaScript代码越来越庞大,越来越复杂(业务逻辑处理或用户交互很多写在前端).网页越来越像桌面程序,需要一个团队分工协作.进度管理. ...
- Javascript模块化编程(三):require.js的用法
Javascript模块化编程(三):require.js的用法 原文地址:http://www.ruanyifeng.com/blog/2012/11/require_js.html 作者: 阮一峰 ...
- Javascript模块化编程(二):AMD规范
Javascript模块化编程(二):AMD规范 作者: 阮一峰 原文地址:http://www.ruanyifeng.com/blog/2012/10/asynchronous_module_d ...
- Javascript模块化编程(二):AMD规范(转)
这个系列的第一部分介绍了Javascript模块的基本写法,今天介绍如何规范地使用模块. (接上文) 七.模块的规范 先想一想,为什么模块很重要? 因为有了模块,我们就可以更方便地使用别人的代码,想要 ...
- Javascript模块化编程(二):AMD规范 作者: 阮一峰
声明:转载自阮一峰的网络日志 这个系列的第一部分介绍了Javascript模块的基本写法,今天介绍如何规范地使用模块. (接上文) 七.模块的规范 先想一想,为什么模块很重要? 因为有了模块,我们就可 ...
- Javascript模块化编程之路——(require.js)
转自:http://www.ruanyifeng.com/blog/2012/10/javascript_module.html Javascript模块化编程(一):模块的写法 随着网站逐渐变成&q ...
随机推荐
- grep命令提示"binary file matches **.log"解决方法
仔细想想,这个问题遇到很多次了,之前一直以为很复杂,一搜索发现解决这么简单,记录一下做备忘. grep test XXX.log Binary file app.log matches 此时使用-a参 ...
- uni-app学习(四)好用的插件2
1. uni-app学习(四)好用的插件2 1.1. 树形结构 点击这里 1.2. 下拉刷新上拉加载组件 如果想把下拉上拉做成自定义的,更加好看,可以使用这个插件 地址这里 举个例子 1.3. 浮动键 ...
- 修复\okldr.mbr ,状态0x000000f
问题: 系统出错显示: 修复\okldr.mbr ,状态0x000000f. 方法: 利用bcdeidt删除错误启动项. bcdeidt的应用 1.启动系统,点击开始,运行中输入cmd,鼠标右键管理员 ...
- Python用Pandas读写Excel
Pandas是python的一个数据分析包,纳入了大量库和一些标准的数据模型,提供了高效地操作大型数据集所需的工具.Pandas提供了大量能使我们快速便捷地处理数据的函数和方法. Pandas官方文档 ...
- 【团队项目3】需求改进&系统设计
一.需求 & 原型改进 1.针对课堂讨论环节老师和其他组的问题及建议,对修改选题及需求进行修改 根据用户反馈,我们针对如下问题做了修改: 问题1:如何保证机构是否是官方的?平台是否有监管? 修 ...
- PWA入门:手把手教你制作一个PWA应用
摘要: PWA图文教程 原文:PWA入门:手把手教你制作一个PWA应用 作者:MudOnTire Fundebug经授权转载,版权归原作者所有. 简介 Web前端的同学是否想过学习app开发,以弥补自 ...
- Lustre 文件系统安装
制作一个本地镜像 reposync configfile: [root@localhost html]# cat lustre-repo.conf [lustre-server] name=lustr ...
- nginx 的 proxy_cache 缓存配置
开头:某个项目涉及到 视频切片下载到本地,然后提供服务给客户端.一个视频有多个文件,存储在本地路径下.当客户端请求的视频在本地没有就会回源, 回源拿到的视频文件再返回给客户端,同时在本地缓存一份,提供 ...
- apache配置文件详解(中英文对照版)
# This is the main Apache server configuration file. It contains the # configuration directives that ...
- Maven中使用tomcat:run出现错误org.eclipse.jdt.internal.compiler.classfmt.ClassFormatException
配置是正常的.查阅资料以后说是jdk版本什么的问题.多方修改没有任何改观.换一个思路去查询tomcat:run怎么运行. 是因为他还是沿用了上一次的tomcat插件(默认是6)所以运行的时候使用 to ...