JavaScript模块化原理浅析

01

写在前面

模块化简单来说就是是指把一个复杂的系统分解到多个模块以方便编码。JS模块化的大致流程为:CommonJS(服务端) -> AMD(浏览器端)-> UMD(兼容了CommonJS和AMD) -> ES Module(ES6标准)。本文将从它们的用法进行介绍,简单实现其原理。并简易实现一个模块化打包工具webpack。

本文将从以下几部分进行总结:

CommonJS的用法及原理

AMD的用法及原理

ES2015标准化

自动化构建

简易实现webpack

02

CommonJS

CommonJS 是一种使用广泛的JavaScript模块化规范,核心思想是通过require方法来同步地加载依赖的其他模块,通过 module.exports 导出需要暴露的接口。

CommonJS的用法

采用CommonJS来进行导入导出的代码用法如下:

// 用require方法导入

constsomeFun=require('./moduleA');

someFun();

// 用module.exports导出

module.exports = someFunc;

CommonJS原理

a.js:

console.log('aaa');

exports.name ='这是a模块的内容';

b.js:

letfs =require('fs');

letpath =require('path');

letb = req('./a.js');

// req即使CommonJS中的require方法

functionreq(mod){

letfilename = path.join(__dirname, mod);

letcontent = fs.readFileSync(filename,'utf8');

/**

* 最后一个参数是函数的内容体,相当于以下函数

*function fn(exports, module, require, __dirname, __filename) {

*  module.exports = '这是另外一个文件导出的内容'

*  return module.exports

*}

*/

letfn =newFunction('exports','require','module','__filename','__dirname', content +'\n return module.exports;');

letmodule= {

exports: {}

};

returnfn(module.exports, req,module, __filename, __dirname);

}

03

AMD

AMD 也是一种 JavaScript 模块化规范,与 CommonJS 最大的不同在于它采用异步的方式去加载依赖的模块。 AMD 规范主要是为了解决针对浏览器环境的模块化问题,最具代表性的实现是 requirejs。

顺便给大家推荐一个裙,它的前面是 537,中间是631,最后就是 707。想要学习前端的小伙伴可以加入我们一起学习,互相帮助。群里每天晚上都有大神免费直播上课,如果不是想学习的小伙伴就不要加啦。

AMD 的优点:

可在不转换代码的情况下直接在浏览器中运行

可加载多个依赖

代码可运行在浏览器环境和 Node.js 环境下

AMD 的缺点:

JavaScript 运行环境没有原生支持 AMD,需要先导入实现了 AMD 的库后才能正常使用。

// 使用define方法定义一个模块

define('a', [],function(){

return'a';

});

define('b', ['a'],function(a){

returna +'b';

});

// 使用require来导入和使用

require(['b'],function(b){

console.log(b);

});

AMD的原理

letfactories = {};

/**

* 实现AMD的define方法

* @param moduleName 模块的名字

* @param dependencies 依赖

* @param factory 工厂函数

*/

functiondefine(modName, dependencies, factory){

factory.dependencies = dependencies;

factories[modName] = factory;

}

/**

* 实现AMD的require方法

* @param mods 引入的模块

* @param callback 回调函数

*/

functionrequire(modNames, callback){

letloadedModNames = modNames.map(function(modName){

letfactory = factories[modName];

letdependencies = factory.dependencies;

letexports;

require(dependencies,function(...dependencyMods){

exports = factory.apply(null, dependencyMods);

});

returnexports;

})

callback.apply(null, loadedModNames);

}

04

ES2015模块化

ES2015 模块化是ECMA提出的JavaScript模块化规范,它在语言的层面上实现了模块化。浏览器厂商和Node.js 都宣布要原生支持该规范。它将逐渐取代CommonJS和AMD规范,成为浏览器和服务器通用的模块解决方案。 采用 ES2015 模块化导入及导出时的代码如下:

// 使用import导入

import{ name }from'./person.js';

// 使用export导出

exportconstname ='musion';

ES2015模块虽然是终极模块化方案,但它的缺点在于目前无法直接运行在大部分 JavaScript 运行环境下,必须通过构建工具转换成标准的 ES5 后才能正常运行。

06

自动化构建

自动化构建就是做这件事情,把源代码转换成发布到线上的可执行 JavaScrip、CSS、HTML 代码,包括如下内容:

顺便给大家推荐一个裙,它的前面是 537,中间是631,最后就是 707。想要学习前端的小伙伴可以加入我们一起学习,互相帮助。群里每天晚上都有大神免费直播上课,如果不是想学习的小伙伴就不要加啦。

代码转换:ECMASCRIPT6 编译成 ECMASCRIPT5、LESS 编译成 CSS 等。

文件优化:压缩 JavaScript、CSS、HTML 代码,压缩合并图片等。

代码分割:提取多个页面的公共代码、提取首屏不需要执行部分的代码让其异步加载。

模块合并:在采用模块化的项目里会有很多个模块和文件,需要构建功能把模块分类合并成一个文件。

自动刷新:监听本地源代码的变化,自动重新构建、刷新浏览器。

代码校验:在代码被提交到仓库前需要校验代码是否符合规范,以及单元测试是否通过。

自动发布:更新完代码后,自动构建出线上发布代码并传输给发布系统。

07

webpack

webpack 是一个打包模块化 JavaScript 的工具,在 webpack 里一切文件皆模块,通过 Loader 转换文件,通过 Plugin 注入钩子,最后输出由多个模块组合成的文件。Webpack 专注于构建模块化项目。

一切文件: JavaScript、CSS、SCSS、图片、模板,在 Webpack 眼中都是一个个模块,这样的好处是能清晰的描述出各个模块之间的依赖关系,以方便 Webpack 对模块进行组合和打包。 经过 Webpack 的处理,最终会输出浏览器能使用的静态资源。

webpack简单揭秘

用webpack来进行打包构建的时候,查看打包📦之后的文件,删掉多余的代码,剩下的核心代码如下:

(function(modules){

functionrequire(moduleId){

varmodule= {

exports: {}

};

// moduleId -> 模块的名字

modules[moduleId].call(module.exports,module,module.exports,require);

returnmodule.exports;

}

returnrequire("./index.js");

}) ({

"./index.js":

(function(module, exports, require){

eval("console.log('hello');\n\n");

})

});

简易实现一个webpack

#! /usr/bin/env node

// 这个文件用来描述如何打包

constpathLib =require('path');

constfs =require('fs');

letejs =require('ejs');

letcwd = process.cwd();

let{ entry,output: { filename, path } } =require(pathLib.join(cwd,'./webpack.config.js'));

letscript = fs.readFileSync(entry,'utf8');

letbundle =`

(function (modules) {

function require(moduleId) {

var module = {

exports: {}

};

modules[moduleId].call(module.exports, module, module.exports, require);

return module.exports;

}

return require("<%-entry%>");

})

({

"<%-entry%>":

(function (module, exports, require) {

eval("<%-script%>");

})

});

`

letbundlejs = ejs.render(bundle, {

entry,

script

});

try{

fs.writeFileSync(pathLib.join(path, filename), bundlejs);

}catch(e) {

console.error('编译失败 ', e);

}

console.log('compile sucessfully!');

依赖其它模块的情况

#! /usr/bin/env node

// 这个文件用来描述如何打包

constpathLib =require('path');

constfs =require('fs');

letejs =require('ejs');

letcwd = process.cwd();

let{ entry,output: { filename, path } } =require(pathLib.join(cwd,'./webpack.config.js'));

letscript = fs.readFileSync(entry,'utf8');

letmodules = [];

script.replace(/require\(['"](.+?)['"]\)/g,function(){

letname =arguments[1];

letscript = fs.readFileSync(name,'utf8');

modules.push({

name,

script

});

});

letbundle =`

(function (modules) {

function require(moduleId) {

var module = {

exports: {}

};

modules[moduleId].call(module.exports, module, module.exports, require);

return module.exports;

}

return require("<%-entry%>");

})

({

"<%-entry%>":

(function (module, exports, require) {

eval(\`<%-script%>\`);

})

<%if(modules.length>0){%>,<%}%>

<%for(let i=0;i

let module = modules[i];%>

"<%-module.name%>":

(function (module, exports, require) {

eval(\`<%-module.script%>\`);

})

<% }%>

});

`

letbundlejs = ejs.render(bundle, {

entry,

script,

modules

});

try{

fs.writeFileSync(pathLib.join(path, filename), bundlejs);

}catch(e) {

console.error('编译失败 ', e);

}

console.log('compile sucessfully!');

08

补充阅读

JavaScript 模块化七日谈:

https://huangxuan.me/2015/07/09/js-module-7day/

JavaScript模块化编程简史(2009-2016):

https://yuguo.us/weblog/javascript-module-development-history/

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,319评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,801评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,567评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,156评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,019评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,090评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,500评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,192评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,474评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,566评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,338评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,212评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,572评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,890评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,169评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,478评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,661评论 2 335

推荐阅读更多精彩内容