JS模块化
1. 不得不说的历史
JS本身就是为了满足简单的页面设计: 页面动画 + 表单提交
 最初并无模块化 or 命名空间的概念
幼年期:无模块化
- 开始需要在页面中加载不同的JS:动画、组件、格式化
- 多种js文件会被分在不同的文件中
- 不同的文件又被同一个模板所引用
  <script src="jquery.js"></script>
  <script src="main.js"></script>
  <script src="dep1.js"></script>
认可:
 文件分离拆分是最基础的模块化(第一步)
- 追问:
 script标签的参数 - async & defer
  <script src="jquery.js" async></script>
总结:
 普通 - 解析到立即阻塞,立刻下载执行当前script 下载到执行js会阻塞渲染
 async - 解析到标签开始异步下载,解析完之后开始执行 执行过程会阻塞主线程渲染
 defer - 解析到标签开始异步下载,下载完成后等待主线程渲染结束再执行
 
 问题出现:
- 污染全局作用域 => 不利于大型项目的开发以及多人团队的共建
成长期:模块化前夜 - IIFE(语法侧的优化)
作用域的把控
例子:
  let count = 0;
  const increase = () => ++count;
  const reset = () => {
    count = 0;
  }
利用函数的块级作用域 - 隔离区
 (() => {
   let count = 0;
   const increase = () => ++count;
   const reset = () => {
    count = 0;
  }
 })();
初步实现了一个最最最最最最简单的模块
尝试定义一个最简单的模块
  const iifeModule = (() => {
    let count = 0;
    const increase = () => ++count;
    const reset = () => {
      count = 0;
    }
    console.log(count);
    increase();
  })();
- 追问:独立模块本身的额外依赖,如何优化
  const iifeModule = ((dependencyModule1, dependencyModule2) => {
    let count = 0;
    const increase = () => ++count;
    const reset = () => {
      count = 0;
    }
    console.log(count);
    increase();
  })(dependencyModule1, dependencyModule2);
** 面试1:了解jquery或者其他很多开源框架的模块加载方案
  const iifeModule = ((dependencyModule1, dependencyModule2) => {
    let count = 0;
    const increase = () => ++count;
    const reset = () => {
      count = 0;
    }
    console.log(count);
    increase();
    return {
      increase, reset
    }
  })(dependencyModule1, dependencyModule2);
  iifeModule.increate();
  iifeModule.increate();
=> 揭示模式 revealing => 上层无需了解底层实现,仅关注抽象 => 框架
- 追问
- 继续模块化横向展开
- 转向框架: jquery|vue|react模块化细节
- 转向设计模式
成熟期:
CJS - Commonjs
- 通过module + exports 去对外暴露接口
- 通过require去引入外部模块
// main.js
const dependencyModule1 = require('./dependencyModule1');
const dependencyModule2 = require('./dependencyModule2');
let count = 0;
const increase = () => ++count;
const reset = () => {
  count = 0;
}
console.log(count);
increase();
exports.increase = increase;
exports.reset = increase;
module.exports = {
  increase, reset
}
// 引入方式
const { increase, reset } = require('./main.js')
// #####################################
// IIFE和CJS复合使用
(function(thisValue, exports, require, module) {
  const dependencyModule1 = require('./dependencyModule1');
  const dependencyModule2 = require('./dependencyModule2'); 
  // 业务逻辑……
}).call(thisValue, exports, require, module);
// 一些开源项目为何要把全局、指针以及框架本身引用作为参数
(function(window, $, undefined) {
  const _show = function() {
    $("#app").val("hi zhaowa");
  }
  window.webShow = _show;
})(window, jQuery);
// 阻断思路
// window - 1. 全局作用域转化成局部作用域,提升执行效率 2. 编译时优化
(function(c){})(window) // window会被优化成c
// jquery - 1. 独立定制复写和挂载 2.防止全局串扰
// undefined - 防止重写
- 缺点:
 针对了服务端的解决方案。异步拉取依赖处理不是很完美
新的问题 —— 异步依赖
AMD规范
新增定义方式:
// define来定义模块
define(id, [depends], callback);
// require进行加载
require([module], callback);
模块定义地方
  define('amdModule', ['dependencyModule1', 'dependencyModule2'], (dependencyModule1, dependencyModule2) => {
    // 业务逻辑……
  })
引入的地方
  require(['amdModule'], amdModule => {
    amdModule.increase();
  })
**面试题: 如果在AMDModule中想兼容已有代码,怎么办?
  define('amdModule', [], require => {
    const dependencyModule1 = require('./dependencyModule1');
    const dependencyModule2 = require('./dependencyModule2');
    // 业务逻辑……
  })
**面试题:手写兼容CJS&AMD
  // 判断关键step1. object还是function step2. exports? step3. define
  (define('amdModule'), [], (require, export, module) => {
    const dependencyModule1 = require('./dependencyModule1');
    const dependencyModule2 = require('./dependencyModule2');
    
    let count = 0;
    const increase = () => ++count;
    const reset = () => {
      count = 0;
    }
    export.increase = increase();
  })(
    // 目标:一次性区分CJS还是AMD
    typeof module === "object"
    && module.exports
    && typeof define !== "function"
      ? // 是CJS
        factory => module.exports = factory(require, exports, module)
      : // AMD
        define
  )
- 缺点:引入成本 很难做到按需加载
CMD
  define('module', (require, exports, module) => {
    let $ = require('jquery');
    // jquery相关逻辑
    let dependencyModule1 = require('./dependencyModule1');
    // dependencyModule1相关逻辑
  })
- 缺点:依赖打包,加载逻辑存在于每个模块中,扩大了模块体积,同时功能上依赖编译
ES6模块化
模块在引入、导出和定义的地方:
  // 引入区域
  import dependencyModule1 from './dependencyModule1';
  import dependencyModule2 from './dependencyModule2';
  // 实现业务逻辑……
  // 导出
  export const increase = () => ++count;
  export const reset = () => {
    count = 0;
  }
  
  export default {
    increase, reset
  }
** 面试:1. 性能 - 按需加载 2.动态模块
 ES11原生解决方案
  import('./esModule.js').then(dynamicModule => {
    dynamicModule.increase();
  })
- 局限性:本质上还是运行时的依赖分析
解决模块化的新思路 - 前端工程化
遗留
根本问题 - 运行时进行依赖分析
解决方案 - 线下执行
追问:可否简述,实现一个编译时依赖处理的思路
<!doctype html>
  <script src="main.js"></script>
  <script>
    // 给构建工具一个标识位
    require.config(__FRAME_CONFIG__);
  </script>
  <script>
    require(['a', 'e'], () => {
      // 业务逻辑
    })
  </script>
</html>
define('a', () => {
  let b = require('b')
  let c = require('c')
})
工程化实现
step1: 扫描依赖关系表
  {
    a:['b', 'c'],
    b: ['d'],
    e: []
  }
step2: 根据依赖关系重制模板
<!doctype html>
  <script src="main.js"></script>
  <script>
    // 构建工具生成数据
    require.config({
      "deps": {
        a:['b', 'c'],
        b: ['d'],
        e: []
      }
    });
  </script>
  <script>
    require(['a', 'e'], () => {
      // 业务逻辑
    })
  </script>
</html>
define('a', () => {
  let b = require('b')
  let c = require('c')
})
step3: 执行工具,采用模块化解决方案处理
  define('a', ['b', 'c'], () => {
    export.run = () => {}
  })
- 构建时生成配置,运行时去运行
- 最终转化成可执行的依赖处理
- 可以拓展
完全体 webpack为核心的前端工程化 + mvvm框架的组件化 + 设计模式
async await & generator
// 1. async await
function wait500(input) {
    return new Promise((resolve, reject) => {
      console.log('wait500', input)
      setTimeout(() => {
        resolve(input + 500)
      }, 500);
    });
}
wait500.then(res => {
  wait500.then(res => {
    wait500.then(res => {
      wait500.then(res => {
      })
    })
  })
})
async function asyncCall() {
  const result = await wait500(0);
  result = await wait500(0);
  result = await wait500(0);
  result = await wait500(0);
  
  console.log('asyncCall', result);
}
asyncCall();
// 2. generator 步进代替then
function* generator() {
  let index = 0;
  while(true) yield index++;
}
let gen = generator();
console.log(gen.next().value);
console.log(gen.next().value);
console.log(gen.next().value);
console.log(gen.next().value);
// 面试重点 - 结合流水线做自动化处理
const GEN_LINE = [1, 2, 3, 4, 5, 6]
(GEN_LINE || []).forEach(it => {
  console.log(gen.next(it).value);
})










