<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>ESModules on Saiga</title>
    <link>http://localhost:1313/tags/esmodules/</link>
    <description>Recent content in ESModules on Saiga</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <managingEditor>wuwenzen@outlook.com (wuwj)</managingEditor>
    <webMaster>wuwenzen@outlook.com (wuwj)</webMaster>
    <lastBuildDate>Sun, 10 Mar 2019 00:00:00 +0000</lastBuildDate>
    <atom:link href="http://localhost:1313/tags/esmodules/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>JavaScript 模块化那些事：从 IIFE 到 ES Modules</title>
      <link>http://localhost:1313/posts/2019-03-10-js-module-patterns/</link>
      <pubDate>Sun, 10 Mar 2019 00:00:00 +0000</pubDate><author>wuwenzen@outlook.com (wuwj)</author>
      <guid>http://localhost:1313/posts/2019-03-10-js-module-patterns/</guid>
      <description>&lt;p&gt;最早写前端的时候，经常是一个 &lt;code&gt;index.html&lt;/code&gt; 里塞一大坨 &lt;code&gt;&amp;lt;script&amp;gt;&lt;/code&gt;，变量到处飞，函数名一改项目全崩。后来慢慢开始接触各种模块化方案，从 IIFE、AMD、CommonJS 一路走到 ES Modules，心态也经历了从凑合能用到必须有规范。&lt;/p&gt;&#xA;&lt;p&gt;这篇就当是给自己的一个小总结：&lt;strong&gt;为什么需要模块化、几种常见写法的区别、现在应该怎么写。&lt;/strong&gt;&lt;/p&gt;&#xA;&lt;hr&gt;&#xA;&lt;h2 id=&#34;为什么要模块化&#34;&gt;为什么要模块化？&lt;/h2&gt;&#xA;&lt;p&gt;几个非常现实的问题：&lt;/p&gt;&#xA;&lt;ul&gt;&#xA;&lt;li&gt;全局变量名冲突：不同文件里不小心定义了同名变量/函数。&lt;/li&gt;&#xA;&lt;li&gt;依赖关系混乱：这个文件依赖谁？要先引哪个 &lt;code&gt;&amp;lt;script&amp;gt;&lt;/code&gt;？&lt;/li&gt;&#xA;&lt;li&gt;代码难以拆分复用：逻辑越写越大，一改就牵一大片。&lt;/li&gt;&#xA;&lt;li&gt;测试困难：无法只「拿出一小块」单独测试。&lt;/li&gt;&#xA;&lt;/ul&gt;&#xA;&lt;p&gt;所以模块化要解决的就是：&lt;/p&gt;&#xA;&lt;blockquote&gt;&#xA;&lt;p&gt;&lt;strong&gt;把代码按职责拆成「小块」 + 明确每块对外暴露什么 + 明确依赖关系。&lt;/strong&gt;&lt;/p&gt;&#xA;&lt;/blockquote&gt;&#xA;&lt;hr&gt;&#xA;&lt;h2 id=&#34;第一阶段iife-模式自执行函数&#34;&gt;第一阶段：IIFE 模式（自执行函数）&lt;/h2&gt;&#xA;&lt;p&gt;当时 jQuery 时代最常见的一种写法，就是用自执行函数「造一个自己的小世界」。&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;&#34;&gt;&lt;code class=&#34;language-js&#34; data-lang=&#34;js&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;(&lt;span style=&#34;color:#66d9ef&#34;&gt;function&lt;/span&gt; () {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#66d9ef&#34;&gt;const&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;API_URL&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;/api/todos&amp;#39;&lt;/span&gt;;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#66d9ef&#34;&gt;function&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;fetchTodos&lt;/span&gt;() {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;return&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;fetch&lt;/span&gt;(&lt;span style=&#34;color:#a6e22e&#34;&gt;API_URL&lt;/span&gt;).&lt;span style=&#34;color:#a6e22e&#34;&gt;then&lt;/span&gt;(&lt;span style=&#34;color:#a6e22e&#34;&gt;res&lt;/span&gt; =&amp;gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;res&lt;/span&gt;.&lt;span style=&#34;color:#a6e22e&#34;&gt;json&lt;/span&gt;());&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  }&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#66d9ef&#34;&gt;function&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;renderTodos&lt;/span&gt;(&lt;span style=&#34;color:#a6e22e&#34;&gt;list&lt;/span&gt;) {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#75715e&#34;&gt;// 渲染逻辑...&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;&lt;/span&gt;  }&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  window.&lt;span style=&#34;color:#a6e22e&#34;&gt;todoApp&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;init&lt;/span&gt;() {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;      &lt;span style=&#34;color:#a6e22e&#34;&gt;fetchTodos&lt;/span&gt;().&lt;span style=&#34;color:#a6e22e&#34;&gt;then&lt;/span&gt;(&lt;span style=&#34;color:#a6e22e&#34;&gt;renderTodos&lt;/span&gt;);&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    },&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  };&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;})();&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;特点：&lt;/p&gt;&#xA;&lt;ul&gt;&#xA;&lt;li&gt;用 IIFE 隔离出一个作用域，避免变量污染全局。&lt;/li&gt;&#xA;&lt;li&gt;需要暴露给外面的东西挂在 &lt;code&gt;window.todoApp&lt;/code&gt; 上。&lt;/li&gt;&#xA;&lt;li&gt;所有代码其实还是在一个文件里，&lt;strong&gt;没有真正意义上的「依赖管理」&lt;/strong&gt;。&lt;/li&gt;&#xA;&lt;/ul&gt;&#xA;&lt;p&gt;适合：&lt;strong&gt;小项目 / 简单封装&lt;/strong&gt;。&lt;br&gt;&#xA;不适合：多页面、多模块的大型项目。&lt;/p&gt;&#xA;&lt;hr&gt;&#xA;&lt;h2 id=&#34;第二阶段amd--requirejs&#34;&gt;第二阶段：AMD / RequireJS&lt;/h2&gt;&#xA;&lt;p&gt;再往后，就出现了 AMD（Asynchronous Module Definition），代表是 RequireJS。&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
