模块化开发与规范标准

400 阅读11分钟

一、模块化开发

模块化开发是当前最重要的前端开发范式。随着前端代码量的激增,可以将不同的代码按照功能或业务划分为不同模块单独去维护,提高开发效率和降低维护成本。模块化只是思想,具体实现需要依赖一些工具。
模块化的演进过程

1. 文件划分方式

缺点:污染全局作用域、命名冲突问题、无法管理模块依赖关系,模块化完全依靠约定。

2. 命名空间方式

文件划分后,为每个文件提供一个命名空间,虽然减少了命名冲突,但仍存在无法管理模块依赖关系问题

// module a 相关状态数据和功能函数
var moduleA = {
  name: 'module-a',
  method1: function () {
    console.log(this.name + '#method1')
  },
  method2: function () {
    console.log(this.name + '#method2')
  }
}

3. IIFE立即执行函数

划分了命名空间,明确了模块的依赖

; (function ($) { // 依赖变量
    // 函数私有作用域
    var name = 'module-a'
    function method1 () {
        console.log(name + '#method1')
    }
    function method2 () {
        console.log(name + '#method2')
    }
    // 挂载到全局对象,实现了私有成员,仅能在闭包中使用
    window.moduleA = {
        method1: method1,
        method2: method2
    }
})(jQuery) // 依赖声明

4. 模块化规范出现

虽然实现了模块的依赖,但模块的加载仍不受控制。于是就出现了一些模块化标准+模块加载器的规范。
CommonJS规范(nodeJS):以同步模式加载模块,不适合浏览器端使用

  • 一个文件就是一个模块
  • 每个模块都有单独的作用域
  • 通过module.exports导出成员
  • 通过require函数载入模块 AMD(Asynchronous Module Definition)异步模块定义(Require.js): 以异步加载模块,同时可以管理模块依赖,适合浏览器端使用(前端模块化演进中妥协的方案)
  • 目前绝大多数的第三方库都支持AMD规范
  • AMD的使用相对复杂(除了业务代码,还要编写一个模块定义与模块依赖相关的代码)
  • 模块划分过于细致后,模块文件的请求频繁且多,导致页面效率底下
// 定义一个模块,'moduleA'为模块名,['jquery', 'moduleB']为依赖项,$, moduleB为依赖项导出成员
define('moduleA', ['jquery', 'moduleB'], function ($, moduleB) {
    return {
        foo: function () {
            $('body').animate({ margin: '100px' })
            moduleB()
        }
    }
});
// 加载一个模块(相当于自动创建一个script标签加载对应的代码)
require(['./moduleA'], function (moduleA) { 
    moduleA.foo()
})

CMD(Common Module Definition)通用模块定义:类似CommonJS规范,但Require.js也兼容了

// CMD规范(类似CommonJS规范)
define(function (require, exports, module) { 
    // 通过require引入依赖
    var $ = require('jquery')
    // 通过exports或者module.exports对外暴露成员
    module.exports = function () { 
        console.log('module 2~')
        $('body').append('<p>module 2</>')
    }
})

5. 模块化标准规范

目前的模块化标准已经确定: image.png ES Modules是ES6中定义的一个标准,因此会存在各种的兼容问题,但随着webpack等打包工具的出现已经可以完全兼容使用了。

二、ES Modules标准

1. 基本特性

    <!-- 通过给script标签添加type = module的属性,就可以以ES Module 的标准执行其中的JS代码 -->
    <script type="module">
        console.log("this is es module")
    </script>
    <!-- 1、ESM自动采用严格模式,忽略'use strict' -->
    <script type="module">
        console.log(this)
    </script>
    <!-- 2、每一个ES Module都是运行在单独的私有作用域中 -->
    <script type="module">
        var foo = 100
        console.log(foo) // 100
    </script>
    <script type="module">
        // console.log(foo) // foo is not defined
    </script>
    <!-- 3、ES Module是通过CORS 的方式请求外部JS模块的,若服务端不支持CORS会出现跨域问题 -->
    <!-- <script type="module" src="https://libs.baidu.com/jquery@3.4.1/dist/jquery.min.js"></script> -->
    <!-- <script type="module" src="https://unpkg.com/jquery@3.4.1/dist/jquery.min.js"></script> -->
    <!-- 4、ES Module的script标签会延迟执行脚本,相当于添加了defer属性,js不会阻塞页面的渲染 -->
    <script defer src="demo.js"></script>
    <p>需要显示的内容</p>

2. ES Modules模块导入导出用法

// 模块导入   ./app.js
import { default as fooName, hello } from './module.js'
console.log(fooName)
console.log(hello()) 

// 模块导出   ./module.js
const foo = 'es modules!'
class Person { }
export {
    foo as default,
    Person
}
  • import导入文件需要提供完整的路径和后缀名,也可以引用CDN地址,使用相对路径的时候./不能省略,使用绝对路径/的时候是从当前文件夹的根目录寻址的。
  • 当仅仅是执行某个模块时,使用import ./module.js加载这个模块并执行它,并不提取模块中内容。
  • 当导入一个模块的所有成员的时候,可以使用import * as mod from './module.js,通过mod对象获取模块内的所有成员。
  • 当导入模块的路径是动态的时候,可以采用import('./module.js')函数,该函数返回的是一个promise对象,在promise对象的then方法中可以拿到模块的所有变量和方法
  • 当模块中导出命名变量和匿名变量时,如:export {name, age}; export default 'default'导出文件,则在导入文件中可以使用import {name, age, default as title} from './module.js'import title, {name, age} from './module.js'
import { name, age } from './module.js'
console.log(name, age) // jack 17
// 模块向外暴露的变量是只读的
name = 'tom'
// 模块向外暴露是引用关系
setTimeout(() => {
    console.log(name, age) // ben 17
}, 1500)

3. ES Modules导入导出成员

一般在模块文件较多的时候,可以在所在文件夹新建一个index.js,然后就可以在此文件中导入导出文件。若加载的模块采用默认导出,则在index.js中相应模块导入的时候需要进行重命名,如:export { default as Button } from './button.js'

// 集中的模块成员导入与导出
// import { Button } from './button.js'
// import { Avartar } from './avatar.js'
// export { Button, Avartar }

export { Button } from './button.js'
export { Avartar } from './avatar.js'

4. 处理ES Module在浏览器环境的兼容(Polyfill)

  <script nomodule src="https://unpkg.com/promise-polyfill@8.1.3/dist/polyfill.min.js"></script>
  <script nomodule src="https://unpkg.com/browser-es-module-loader@0.4.1/dist/babel-browser-build.js"></script>
  <script nomodule src="https://unpkg.com/browser-es-module-loader@0.4.1/dist/browser-es-module-loader.js"></script>
  <script type="module">
    import { foo } from './module.js'
    console.log(foo)
  </script>

在浏览器环境不支持使用ES Module时,需要使用browser-es-module-loader处理ES Module模式加载的代码。但为了避免在支持使用ES Module的浏览器环境中二次执行模块内代码,可以使用nomodule。含义是:只有当前浏览器不支持ES Module时,采取加载script标签引用的js文件

5. ES Module在node环境的基本使用及支持情况

支持情况: node8.5+之后,内部就已经以实验特性的方式支持ES Module
使用步骤:

  • 将文件的扩展名由 .js 改为 .mjs
  • 启动时需要额外添加 --experimental-modules 参数,表示启用``ES Module`实验特性
// 我们也可以通过 esm 加载内置模块了
import fs from 'fs'
fs.writeFileSync('./foo.txt', 'es module working')

// 也可以直接提取模块内的成员,内置模块兼容了 ESM 的提取成员方式
import { writeFileSync } from 'fs'
writeFileSync('./bar.txt', 'es module working')

// 对于第三方的 NPM 模块也可以通过 esm 加载
import _ from 'lodash'
_.camelCase('ES Module')

// 不支持,因为第三方模块都是导出默认成员
// import { camelCase } from 'lodash'
// console.log(camelCase('ES Module'))

6. ES Module in Node.js

image.png 要点一:

  • ES Modules中可以导入CommonJS模块
  • CommonJS中不能导入ES Modules模块
  • CommonJS中始终只会导出一个默认成员
  • 注意import不是解构导出对象 要点二(使用差异): image.png 要点三(node12+新版本):
  • 当在package.json中添加type = module后,项目中默认使用ES Module规范,文件后缀可以以.js结尾,但如果想使用CommonJS规范时需要将文件后缀名修改为.cjs。 要点四(node8.5以前版本):
  • 可以使用Bable进行兼容,执行yarn add @babel/core @babel/node @babel/preset-env --dev
  • 需要babel的插件转换,preset-env包含了各种插件

三、Webpack与Rollup打包工具

1、ES Modules模块化的弊端

  • 存在环境兼容问题
  • 模块文件过多,网络请求频繁
  • 不仅仅是JS,所有的前端资源都需要模块化 模块化打包工具可以针对整个前端应用进行模块化,目前主流的的打包工具是WebpackRollup

2、webpack使用

  • webpack的工作模式: nodedevelopmentproduction三种
  • webpack打包结果运行原理:将所有的模块放到同一个文件中,通过一些基础代码维护模块之间的依赖关系
  • LoaderWebpack的核心特性,借助Loader就可以加载任何类型的资源
  • webpack建议在代码中引入任何当前代码需要的资源,即动态导入。因为真正需要资源的不是应用,而是代码。
  • webpack的哲学:Javascript驱动整个前端应用,资源文件动态导入,确保上线资源不缺少,都是必要的。 启发:关注新事物的思想才是技术水平提升的重点。
  1. 文件加载器file-loader
    图解: image.png

  2. URL加载器url-loader
    使用要点:小文件使用Data URLs,减少请求次数。大文件单独提取存放,提高加载速度(临界大小10KB) 图解:Data URLs image.png 例如:
    文本url: data:text/html;charset=UTF-8,<h1>html content</h1> 图片url: ...SUowEC

  3. 编译加载器babel-loader

  • webpack只是打包工具
  • 加载器可以用来编译转换代码
  1. 加载器分类
  • 编译转换类:例如css-loadercss代码转换为bundle.js中的一个模块
  • 文件操作类:例如file-loader将资源文件拷贝到输出目录,并在bundle.js中导出文件访问路径。
  • 代码检查类:例如eslint-loader检验代码的风格,从而使代码质量有所提升。
  1. 加载资源的方式
  • 遵循ESModules标准的import声明(JavaScript代码加载推荐统一使用此种)
import createHeading from './heading.js'
import icon from './icon.png'
import './main.css'
  • 遵循CommonJS标准的require函数
const createHeading = require('./heading.js').default
const icon = require('./icon.png')
require('./main.css')
  • 遵循AMD标准的define函数和require函数
// 模块声明
definne(['./heading.js', './icon.png', './main.css'], (createHeading, icon) => { 
    const heading = createHeading.default()
    const img = new Image()
    img.src = icon
    document.body.append(heading)
    document.body.append(img)
})
// 模块引入
require(['./heading.js', './icon.png', './main.css'], (createHeading, icon) => { 
    const heading = createHeading.default()
    const img = new Image()
    img.src = icon
    document.body.append(heading)
    document.body.append(img)
})
  • 样式代码中的@import指令和url函数
  • HTML代码中图片标签的src属性,a标签的href属性
{
    test: /.html$/,
    use: {
        loader: 'html-loader',
        options: {
            attrs: ['img:src', 'a:href']
        }
    }
}
  1. 核心工作原理 图解:webpack通过入口文件遍历整个文件加载书,然后将不同的资源文件交由相应的loader加载此模块,最后将结果放到bundle.js中,从而实现整个项目的打包,因此Loader机制是webpack的核心。 image.png

  2. Loader的工作原理

  • wepackLoader是一个函数,导出的一段JavaScript代码。
  • Loader负责资源文件从输入到输出的转换
  • 对于同一个资源可以依次使用多个loader处理
// 自定义Loader
const marked = require('marked')
module.exports = source => { 
    // console.log(source)
    const html = marked(source)
    // 返回一段js代码
    // return `export default ${JSON.stringify(html)}`
    return html
}
  1. 插件机制
    Loader目的:专注实现资源模块加载
    插件目的:增强webpack的自动化能力(如:清除文件目录、拷贝静态文件到输出目录)
  • clean-wabpack-plugin: 自动清除输出目录插件
  • html-wabpack-plugin:自动生成使用bundle.jsHTML,每个HtmlWebpackPlugin负责生成一个html入口文件
  • copy-wabpack-plugin:文件拷贝功能 webpack的中有许多的钩子,代表着不同的执行阶段,我们可以在钩子上挂载任务扩展webpack插件的能力。 image.png 自定义插件:通过在生命周期的钩子中挂载函数实现扩展
class MyPlugin {
    apply (compiler) {
        console.log('自定义插件')
        // tap方法注册钩子函数(emit是其中一个钩子)
        compiler.hooks.emit.tap('MyPlugin', compilation => {
            // compilation可以理解为此次打包的上下文
            for (const name in compilation.assets) {
                if (name.endsWith('.js')) {
                    const contents = compilation.assets[name].source()
                    const withoutComments = contents.replace(/\/\*\*+\*\//g, '')
                    compilation.assets[name] = {
                        source: () => withoutComments,
                        size: () => withoutComments.length
                    }
                }
            }
        })
    }
}

module.exports = {
    plugins: [ new MyPlugin() ]
}

9、开发体验优化

  • 自动编译 + 代理API + 自动唤起浏览器 + 自动刷新浏览器 + HMR: Webpack Dev Server集成了这些功能
  • 其中静态资源的访问:Webpack Dev Server配置devServer: { contentBase: './public' }
  • 配置Source Map模式: eval-是否使用eval执行模块代码;cheap-Source Map是否包含行信息;module-是否能够得到Loader处理之前的源代码。开发阶段一般选择cheap-module-eval-source-map模式,生产环境一般选择nonenosources-source-map模式
  • 配置HMR(Hot Module Replacement)模块热替换:module.hot.accept
  • Tree Shaking: 生产环境默认开启Tree Shaking, 开发环境可以通过配置。Tree Shaking的前提是通过ES Modules的模式组织代码。最新版的babel-loader并不会导致Tree Shaking失效。
    optimization: { // webpack内部的优化配置
        // usedExports负责标记[枯树叶]
        usedExports: true,
        // minimize负责[摇掉]他们
        minimize: true
    }
  • 合并模块:尽可能将所有的模块合并输出到一个函数中,既提升了运行效率,又减少了代码体积。该特性又被称为Scope Hoisting作用域提升。
    optimization: { // webpack内部的优化配置
        concatenateModules: true,
    }
  • 标记sideEffects(副作用):模块执行时,除了导出成员之外所作的事情。一般用于npm包标记是否有副作用。生产环境默认开启sideEffects
  • Code Splitting代码分割:分包、按需加载、模块打包有必要。原因:http1.1同域并行请求限制、每次请求有一定的延迟、请求的Header浪费带宽流量。方法:多入口打包、动态导入
  • Multi Entry多入口打包:适用于多页应用程序,一个页面对应一个打包入口,公共部分单独提取。
  • Split Chunks提取公共模块
    optimization: { // webpack内部的优化配置
        splitChunks: {
          // 自动提取所有公共模块到单独 bundle
          chunks: 'all'
        }
    }
  • Dynamic Imports动态导入:需要用到某个模块时,再加载这个模块,动态导入的模块会被自动分包。
  • Magic Comments魔法注释:import(/* webpackChunkName: 'components' */'./posts/posts')
  • MiniCssExtractPlugin提取CSS到单独的文件:实现CSS模块的按需加载(CSS体积超过150KB后使用比较合适)
  • OptimizeCssAssetsWebpackPlugin压缩输出的CSS文件:
    optimization: {
        minimizer: [
            new TerserWebpackPlugin(), // webpack默认的JS压缩插件
            new OptimizeCssAssetsWebpackPlugin() // 压缩输出的`CSS`文件
        ]
    },
  • 输出文件名Hash:当启用服务器的静态资源缓存时,只有Hash名变更时,才会重新请求。
  output: {
    filename: '[name]-[contenthash:8].bundle.js'
  }

3、Rollup使用

  1. 区别于webpackRollup仅仅是一款ESM打包器。默认会开启Tree Shaking
  2. Rollup支持使用插件的方式扩展,插件是Rollup唯一的扩展方式。
  3. Rollup默认只会处理ES Modules模块,若要支持加载CommonJS模块可以采用rollup-plugin-commonjs。因为有需要的npm包采用的CommonJS模式组织的代码。
  4. Dynamic Imports动态导入,代码拆分:
import('./logger').then(({ log }) => {
    log('code splitting~')
})
  1. 多入口打包
    input: {
        foo: 'src/index.js',
        bar: 'src/album.js',
    }

总结Rollup使用优缺点:

  • 输出结果更加扁平
  • 自动移除未引用代码
  • 打包结果依然完全可读
  • 加载非ESM的第三方模块比较复杂
  • 模块最终都被打包到一个函数中,无法实现HMR
  • 浏览器环境中,代码拆分功能依赖AMD

4、Parcel使用

零配置的前端应用打包。

四、规范化标准

  1. 为什么要有规范标准?
  • 软件开发需要多人协作
  • 不同的开发者具有不同的编码习惯和喜好
  • 不同的喜好增加了项目维护成本
  • 每个项目或团队都需要明确统一的标准
  1. 哪里需要规范化标准?
  • 代码、文档、甚至提交日志
  • 开发过程中人为编写的成果物
  • 代码标准化规范最为重要,影响着代码的质量和维护成本
  1. 常见的规范化实现方式
  • eslintstylelint
  • eslint 检查typescript
  • stylelint使用
  • stylelint使用
  • eslint结合git hooks (Husky)