经历过杂乱的js函数式编程的人在认识了模块化和模块加载器之后,一定觉得是一种福音。模块化让我们更加有组织有模块地去编写我们的代码,模块化加载器让我们更加方便和清晰地进行模块定义和依赖管理。现在主要的模块化规范是commonJS,AMD和CMD,commonJS主要是用于node服务端,AMD和CMD主要用于浏览器端,代表框架分别是requireJS和seaJS;作为前端,当然更熟悉的是requireJS和seaJS,但是对于我个人而言,commonJS的编码方式我更喜欢,因为简单,无需使用define包装。
如今,要特别感谢前端工程化的出现,让commonJS的编码方式在前端变成可能,比如我们熟悉的Browserify,当然作为国内最强大的前端工程化工具——fis,当然也对模块化也有自己的实现。下面我们来学习一下fis3是如何实现模块化构建的。当然你也可以直接阅读官方文档: fis3模块化
模块化框架一般包含了模块的依赖分析、模块加载并保持依赖顺序等功能。但在 FIS 中,依赖本身在构建过程中就已经分析完成,并记录在静态资源映射表中,那么对于线上运行时,模块化框架就可以省掉依赖分析这个步骤了。
fis3中针对前端模块化框架的特性自动添加define包装,以及根据配置生成对应的require依赖标识主要是通过对应的模块化插件实现的:
fis3-hook-commonjs
fis3-hook-amd
fis3-hook-cmd
生成了规范的模块文件之后,如何将模块之间的依赖关系生成静态资源映射表,则是通过
fis3-postpackager-loader 。
这个插件用于分析页面中使用的和依赖的资源(js或css), 并将这些资源做一定的优化后插入页面中。
下面我们结合栗子来学习一下这些模块化插件是如何工作的。先看看我们专题页项目的目录结构!
static/ #项目静态文件目录 common/ #公共静态文件目录 js/ lib/ #类库文件 mod.js require.js sea.js mod/ #需要模块化的文件 react.js jquery.js css/ #css文件目录 style.css images/ #图片文件目录 style.png commponents/ #公共组件,也是需要模块化加载的 HelloMessage/ HelloMessage.jsx HelloMessage.css helloworld/ #简单的例子 index.html index.css index.jsx fis-conf.js #fis配置文件 package.json #包配置文件
在浏览器环境运行的代码,如果我们希望采用commonJS规范作为模块化开发,则需要安装 fis3-hook-commonjs 插件, npm install fis3-hook-commonjs --save
,还要配合 mod.js 来使用;
安装完成之后看一下fis-conf.js如何配置:
###fis-conf.js /*设置编译范围*/ fis.set('project.files', ['static/**']); /*设置发布路径*/ fis.match(///static//(.*)/i, { release: '/staticPub/$1', /*所有资源发布时产出到 /staticPub 目录下*/ url: '/staticPub/$1' /*所有资源访问路径设置*/ }); /*指定模块化插件*/ fis.hook('commonjs', { paths: { jquery: '/static/common/js/mod/jquery', //设置jquery别名 react: '/static/common/js/mod/react' //设置react别名 } }); /*指定哪些目录下的文件执行define包裹*/ fis.match('/static/common/js/mod/**', { isMod: true }); fis.match('/static/common/components/**', { isMod: true }); fis.match('/static/helloworld/**', { isMod: true }); /*模块化加载器配置*/ fis.match('::package', { postpackager: fis.plugin('loader', { allInOne: true, //js&css打包成一个文件 sourceMap: true, //是否生成依赖map文件 useInlineMap: true //是否将sourcemap作为内嵌脚本输出 }) }); /*支持react*/ fis.match('*.jsx', { rExt: '.js', parser: fis.plugin('react', {}) });
注意:需要对目标文件设置 isMod 属性,说明这些文件是模块化代码。这样才会被自动加上define包装,才能在浏览器里面运行。 fis3-postpackager-loader 的作用则是分析这些文件的依赖关系并生成对应的sourceMap文件,让mod.js分析并加载模块对应的文件到浏览器中。
#helloworld/index.html <!DOCTYPE html> <html> <head> <title>繁星网 | 全球最大音乐现场直播平台</title> <meta charset="utf-8"> <link rel="stylesheet" type="text/css" href="/static/common/css/style.css"> <link rel="stylesheet" type="text/css" href="./css/index.css"> </head> <body> <div id="helloApp"></div> </body> <script type="text/javascript" src="/static/common/js/lib/mod.js"></script> <script type="text/javascript"> require(['./index']);//异步加载index.js模块 </script> </html> #helloworld/index.jsx //引入React和HelloMessage模块 var React = require('react'); var HelloMessage = require('/static/common/components/HelloMessage/HelloMessage.react'); React.render( <HelloMessage message="I like commonjs!" />, document.getElementById('helloApp') ); #common/components/HelloMessage/HelloMessage.react.jsx var React = require('react'); var HelloMessage = React.createClass({ render: function() { return ( <h1>Hello, {this.props.message}</h1> ); } }); module.exports = HelloMessage;
helloworld/index.html需要引入mod.js作为模块化加载器,然后通过require([./index])异步加载index模块;
helloworld/index.jsx依赖React和HelloMessage模块,写法就是我们熟悉的commonJS的方式;
common/components/HelloMessage/index.jsx就是HelloMessage模块,它也依赖React模块;
从上面的jsx文件我们可以轻易地发现,不管是react还是jsx文件都没有任何define包装,写法就commonJS一模一样,但是这样在浏览器肯定是跑不起来的,还需要fis帮我们构建模块包装和依赖分析。OK,一切准备就绪,我们就开始执行fis脚本:
fis3 release -d ./
我们来看看staticPub目录下面产出的编译文件:
#helloworld/index.html <!DOCTYPE html> <!DOCTYPE html> <html> <head> <title>繁星网 | 全球最大音乐现场直播平台</title> <meta charset="utf-8"> <link rel="stylesheet" type="text/css" href="/staticPub/helloworld/index.html_aio.css" /> </head> <body> <div id="helloApp"></div> <script type="text/javascript" src="/staticPub/common/js/lib/mod.js"></script> <script type="text/javascript">/*resourcemap*/ require.resourceMap({ "res": { "static/common/js/mod/react": { "url": "/staticPub/common/js/mod/react.js", "type": "js" }, "static/common/components/HelloMessage/HelloMessage.react": { "url": "/staticPub/common/components/HelloMessage/HelloMessage.react.js", "type": "js", "deps": [ "static/common/js/mod/react" ] }, "static/helloworld/index": { "url": "/staticPub/helloworld/index.js", "type": "js", "deps": [ "static/common/js/mod/react", "static/common/components/HelloMessage/HelloMessage.react" ] } }, "pkg": {} }); require(['static/helloworld/index']);//异步加载index.js模块 </script> </body> </html>
我们来看看有哪些变化:
1、index.html中css文件被打包成一个
<link rel="stylesheet" type="text/css" href="/static/common/css/style.css">
<link rel="stylesheet" type="text/css" href="./css/index.css">
变成了一个
<link rel="stylesheet" type="text/css" href="/staticPub/helloworld/index.html_aio.css" />
2、上面的index.html多了一份sourceMap脚本;
这是因为在fis3-postpackager-loader的配置中加了useInlineMap:true,可以阅读 文档 了解更多配置。
我们再来看看helloworld/index.jsx和HellowMessage/HelloMessage.react.jsx的变化:
#hellowrold/index.js define('static/helloworld/index', function(require, exports, module) { //引入React和HelloMessage模块 var React = require('static/common/js/mod/react'); var HelloMessage = require('static/common/components/HelloMessage/HelloMessage.react'); React.render( React.createElement(HelloMessage, {message: "I like commonjs!"}), document.getElementById('helloApp') ); }); #common/components/HelloMessage/HelloMessage.react.js define('static/common/components/HelloMessage/HelloMessage.react', function(require, exports, module) { var React = require('static/common/js/mod/react'); var HelloMessage = React.createClass({displayName: "HelloMessage", render: function() { return ( React.createElement("h1", null, "Hello, ", this.props.message) ); } }); module.exports = HelloMessage; }); #common/js/mod/react.js define('static/common/js/mod/react', function(require, exports, module) { //react code... }
1、所有的.jsx变成了.js文件,这是fis3-parser-react插件做的;
2、js文件都加了define包装,比如"static/helloworld/index"是index模块的moduleId;
3、require('react')编译成了require('static/common/js/mod/react'),因为我们通过path配置了别名;
我们可以发现,通过fis生成的js代码define的moduleId跟index.html中sourceMap的moduleId是一致的。这样mod.js就能通过resourceMap的依赖关系加载到所有的模块啦!下面是demo在浏览器中的运行结果截图:以上就是通过fis3-hook-commonjs实现模块化的过程,当然插件还有一些配置项供开发人员配置,感兴趣的同学可以通过阅读 fis3-hook-commonjs 的文档自行了解。
首先安装 fis3-hook-amd 插件, npm install fis3-hook-amd --save
。 如果我们理解fis3-hook-commonjs的使用方式,换成fis3-hook-amd就很简单,使用方式的唯一的不同就是hook的插件由commonjs变为amd:
fis.hook('amd', { paths: { jquery: '/static/common/js/mod/jquery', react: '/static/common/js/mod/react' } });
当然此时我们的模块化框架要用require.js啦!所以index.html我们要把mod.js换成require.js。
<script type="text/javascript" src="/static/common/js/lib/require.js"></script>
fis3-release -d ./
下面我们看看编译之后的产出文件:
#helloworld/index.html <!DOCTYPE html> <html> <head> <title>繁星网 | 全球最大音乐现场直播平台</title> <meta charset="utf-8"> <link rel="stylesheet" type="text/css" href="/staticPub/helloworld/index.html_aio.css" /> </head> <body> <div id="helloApp"></div> <script type="text/javascript" src="/staticPub/common/js/lib/require.js"></script> <script type="text/javascript">/*resourcemap*/ require.config({paths:{ "static/common/js/mod/jquery": "/staticPub/common/js/mod/jquery", "static/common/js/mod/react": "/staticPub/common/js/mod/react", "static/common/components/HelloMessage/HelloMessage.react": "/staticPub/common/components/HelloMessage/HelloMessage.react", "static/helloworld/index": "/staticPub/helloworld/index" }}); require(['static/helloworld/index']);//异步加载index.js模块 </script> </body> </html> #helloworld/index.js define('static/helloworld/index', ['require', 'exports', 'module', 'static/common/js/mod/react', 'static/common/components/HelloMessage/HelloMessage.react'], function(require, exports, module) { //引入React和HelloMessage模块 var React = require('static/common/js/mod/react'); var HelloMessage = require('static/common/components/HelloMessage/HelloMessage.react'); React.render( React.createElement(HelloMessage, {message: "I like AMD!"}), document.getElementById('helloApp') ); }); #common/components/HelloMessage/HelloMessage.js define('static/common/components/HelloMessage/HelloMessage.react', ['require', 'exports', 'module', 'static/common/js/mod/react'], function(require, exports, module) { var React = require('static/common/js/mod/react'); var HelloMessage = React.createClass({displayName: "HelloMessage", render: function() { return ( React.createElement("h1", null, "Hello, ", this.props.message) ); } }); module.exports = HelloMessage; });
注意,index.html内嵌脚本生成的sourceMap变成下面的格式,因为是AMD规范嘛:
require.config({paths:{ "static/common/js/mod/jquery": "/staticPub/common/js/mod/jquery", "static/common/js/mod/react": "/staticPub/common/js/mod/react", "static/common/components/HelloMessage/HelloMessage.react": "/staticPub/common/components/HelloMessage/HelloMessage.react", "static/helloworld/index": "/staticPub/helloworld/index" }});
js文件也被包装成了遵循AMD规范的define形式。下面是demo执行结果:
安装 fis3-hook-cmd 插件, npm install fis3-hook-cmd --save
。 该fis-conf.js配置文件:
/*指定模块化插件*/ fis.hook('cmd', { paths: { jquery: '/static/common/js/mod/jquery', react: '/static/common/js/mod/react' } });
改index.html模块加载器:
<script type="text/javascript" src="/static/common/js/lib/sea.js"></script>
异步加载入口index模块改为:
seajs.use(['./index']);//异步加载index.js模块
fis3-release -d ./
注意:运行完成之后你会发现程序无法运行,因为react模块找不到,为什么呢?一般情况下,我们下载的开源框架都自己实现了amd包装,比如react的源码:
/** * React v0.13.0 */ (function(f) { if (typeof exports === "object" && typeof module !== "undefined") { module.exports = f() //注意看这里,这就是默认是用amd } else if (typeof define === "function" && define.amd) { define([], f) } else { var g; if (typeof window !== "undefined") { g = window } else if (typeof global !== "undefined") { g = global } else if (typeof self !== "undefined") { g = self } else { g = this } g.React = f() } })(function() { var define, module, exports; //这里才是react的内部实现,源码会返回一个React对象 return React; });
对于这类框架fis3-hook-amd会识别define.amd并将define([], f)替换成define('static/common/js/mod/react', [], f),但是我们运行fis3-hook-cmd就无法识别了,所以就无法通过define定义模块,define([], f)不会有任何变化。我们把define.amd改成define.cmd再运行一下fis就会发现了define([], f)变成了define('static/common/js/mod/react', [], f)。
再看看编译之后的产出文件:
#helloworld/index.html <!DOCTYPE html> <html> <head> <title>繁星网 | 全球最大音乐现场直播平台</title> <meta charset="utf-8"> <link rel="stylesheet" type="text/css" href="/staticPub/helloworld/index.html_aio.css" /> </head> <body> <div id="helloApp"></div> <script type="text/javascript" src="/staticPub/common/js/lib/sea.js"></script> <script type="text/javascript">/*resourcemap*/ seajs.config({alias:{ "static/common/js/mod/react": "/staticPub/common/js/mod/react", "static/common/components/HelloMessage/HelloMessage.react": "/staticPub/common/components/HelloMessage/HelloMessage.react", "static/helloworld/index": "/staticPub/helloworld/index" }}); // require(['./index']);//异步加载index.js模块 seajs.use(['static/helloworld/index']);//异步加载index.js模块 </script> </body> </html> #helloworld/index.js define('static/helloworld/index', ['static/common/js/mod/react', 'static/common/components/HelloMessage/HelloMessage.react'], function(require, exports, module) { //引入React和HelloMessage模块 var React = require('static/common/js/mod/react'); var HelloMessage = require('static/common/components/HelloMessage/HelloMessage.react'); React.render( React.createElement(HelloMessage, {message: "I like CMD!"}), document.getElementById('helloApp') ); }); #common/components/HelloMessage/HelloMessage.js define('static/common/components/HelloMessage/HelloMessage.react', ['static/common/js/mod/react'], function(require, exports, module) { var React = require('static/common/js/mod/react'); var HelloMessage = React.createClass({displayName: "HelloMessage", render: function() { return ( React.createElement("h1", null, "Hello, ", this.props.message) ); } }); module.exports = HelloMessage; });
再来看看index.html内嵌脚本生成的sourceMap:
seajs.config({alias:{ "static/common/js/mod/react": "/staticPub/common/js/mod/react", "static/common/components/HelloMessage/HelloMessage.react": "/staticPub/common/components/HelloMessage/HelloMessage.react", "static/helloworld/index": "/staticPub/helloworld/index" }});
查看结果:
因为工程化,让模块化变得简单,可复用!你不用在乎使用你模块的人是使用commonJS还是seaJS还是requireJS作为模块加载器,你只需要专心开发你的模块,并通过require加载你要依赖的模块即可。怎么样?是不是很爽?那就用起来吧~ 有兴趣的同学可以看一下demo: fis3-mudule-demo