你不容错过的JavaScript高级语法(模块化)

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 你不容错过的JavaScript高级语法(模块化)

众所周知,js在前端开发中的地位。学好它,真的很重要。


下面这篇文章,介绍一下模块化。


什么是模块化?


到底什么是模块化、模块化开发呢?


  • 事实上模块化开发最终的目的是将程序划分成一个个小的结构。


  • 这个结构中编写属于自己的逻辑代码,有自己的作用域,不会影响到其他的结构。


  • 这个结构可以将自己希望暴露的变量、函数、对象等导出给其结构使用。


  • 也可以通过某种方式,导入另外结构中的变量、函数、对象等。 上面说提到的结构,就是模块;按照这种结构划分开发程序的过程,就是模块化开发的过程。


模块化的历史


在网页开发的早期,Brendan Eich开发JavaScript仅仅作为一种脚本语言,做一些简单的表单验证或动画实现等,那个时候代码还是很少的:


  • 这个时候我们只需要讲JavaScript代码写到


  • 并没有必要放到多个文件中来编写。


但是随着前端和JavaScript的快速发展,JavaScript代码变得越来越复杂了:


  • ajax的出现,前后端开发分离,意味着后端返回数据后,我们需要通过JavaScript进行前端页面的渲染。


  • SPA的出现,前端页面变得更加复杂:包括前端路由、状态管理等等一系列复杂的需求需要通过JavaScript来实现。


  • 包括Node的实现,JavaScript编写复杂的后端程序,没有模块化是致命的硬伤。 所以,模块化已经是JavaScript一个非常迫切的需求。所以ES6(2015)才推出了自己的模块化方案。


在此之前,为了让JavaScript支持模块化,涌现出了很多不同的模块化规范:AMD、CMD、CommonJS等。


没有模块化带来的问题


  • 比如命名冲突的问题。 通过立即函数调用表达式(IIFE)来解决上面的问题。因为函数有自己的作用域,不会造成不同文件命名冲突。


// a.js
    var moduleA = (function() {
      var name = "llm"
      var age = 22
      var isFlag = true
      return {
        name: name,
        isFlag: isFlag
      }
    })()


// b.js
    var moduleB = (function() {
      var name = "zh"
      var isFlag = false
      return {
        name: name,
        isFlag: isFlag
      }
    })()


// 使用
    moduleA.name
    moduleB.name


但是,我们其实带来了新的问题:


  • 我必须记得每一个模块中返回对象的命名,才能在其他模块使用过程中正确的使用。


  • 代码写起来混乱不堪,每个文件中的代码都需要包裹在一个匿名函数中来编写。


  • 在没有合适的规范情况下,每个人、每个公司都可能会任意命名、甚至出现模块名称相同的情况。 所以,我们会发现,虽然实现了模块化,但是我们的实现过于简单,并且是没有规范的。


我们需要制定一定的规范来约束每个人都按照这个规范去编写模块化的代码。这个规范中应该包括核心功能:模块本身可以导出暴露的属性,模块又可以导入自己需要的属性。JavaScript社区为了解决上面的问题,涌现出一系列好用的规范,接下来我们就学习具有代表性的一些规范。


CommonJS规范和Node


我们需要知道CommonJS是一个规范,最初提出来是在浏览器以外的地方使用,并且当时被命名为ServerJS,后来为了体现它的广泛性,修改为CommonJS,平时我们也会简称为CJS。


  • Node是CommonJS在服务器端一个具有代表性的实现。


  • Browserify是CommonJS在浏览器中的一种实现。


  • webpack打包工具具备对CommonJS的支持和转换。


所以,Node中对CommonJS进行了支持和实现,让我们在开发node的过程中可以方便的进行模块化开发。


  • 在Node中每一个js文件都是一个单独的模块。


  • 这个模块中包括CommonJS规范的核心变量:exports、module.exports、require。


  • 我们可以使用这些变量来方便的进行模块化开发。 前面我们提到过模块化的核心是导出和导入,Node中对其进行了实现:


  • exports和module.exports可以负责对模块中的内容进行导出。


  • require函数可以帮助我们导入其他模块(自定义模块、系统模块、第三方库模块)中的内容。


Node.js模块化


Node中对CommonJS进行了支持和实现,让我们在开发node的过程中可以方便的进行模块化开发:


  • 在Node中每一个js文件都是一个单独的模块。


  • 这个模块中包括CommonJS规范的核心变量:exports、module.exports、require


  • exportsmodule.exports可以负责对模块中的内容进行导出。


  • require函数可以帮助我们导入其他模块(自定义模块、系统模块、第三方库模块)中的内容。


下面我们将来介绍exports、module.exports、require的使用。


  • exports是一个对象,我们可以在这个对象中添加很多个属性,添加的属性会导出。


  • 我们也可以通过module.exports直接导出一个对象。


  • 我们通过require()函数导入一个文件。并且该文件导出的变量。


下面来详细介绍一个module.exports


  • CommonJS中是没有module.exports的概念的。


  • 但是为了实现模块的导出,Node中使用的是Module的类,每一个模块都是Module的一个实例,也就是module。


  • 所以在Node中真正用于导出的其实根本不是exports,而是module.exports。


  • 因为module才是导出的真正实现者。


  • 并且内部将exports赋值给module.exports


该方式的导入导出有以下特点:


  • Node中的文件都运行在一个函数中。可以通过打印console.log(arguments.callee + "")来验证。


网络异常,图片无法展示
|


  • 导入导出是值的引用,如果导出的是一个基本数据类型值,那么导出文件改变该值,然后导入文件该变量的值也不会变。


// a.js
    const obj = require("./b.js")
    console.log(obj)
    setTimeout(() => {
      obj.name = "llm"
    }, 1000)


// b.js
    const info = {
      name: "zh",
      age: 22,
      foo: function() {
        console.log("foo函数~")
      }
    }
    setTimeout(() => {
      console.log(info.name) // llm
    }, 2000)
    module.exports = info


  • 他是通过require 函数来导入的,只有在执行js代码才会知道模块的依赖关系。


  • 代码是同步执行的。


  • 模块多次引入,只会加载一次。每个module内部会存在一个loaded来确定是否被加载过。


  • 代码循环引入的时候,深度优先来加载模块。然后再广度优先。


下面来详细介绍一个require的导入细节


我们现在已经知道,require是一个函数,可以帮助我们引入一个文件(模块)中导出的对象。


那么,require的查找规则是怎么样的呢?


详细查找规则,请访问这里


这里我总结比较常见的查找规则:导入格式如下:require(X)


  • X是一个Node核心模块,比如path、http。


  • 直接返回核心模块,并且停止查找。


  • X是以 ./..//(根目录)开头的。


  • 第一步:将X当做一个文件在对应的目录下查找。


  • 如果有后缀名,按照后缀名的格式查找对应的文件。


  • 如果没有后缀名,会按照如下顺序:


  • 直接查找文件X


  • 查找X.js文件


  • 查找X.json文件


  • 查找X.node文件


  • 第二步:没有找到对应的文件,将X作为一个目录


  • 查找目录下面的index文件


  • 查找X/index.js文件


  • 查找X/index.json文件


  • 查找X/index.node文件


  • 如果没有找到,那么报错:not found


  • 直接是一个X(没有路径),并且X不是一个核心模块。那么它将向上级目录查找node_modules文件。直到查找到根目录下。我们可以通过module.paths打印出这些查找目录。如果上面的路径中都没有找到,那么报错:not found。


模块的加载细节


  • 模块在被第一次引入时,模块中的js代码会被运行一次


  • 模块被多次引入时,会缓存,最终只加载(运行)一次 为什么只会加载运行一次呢?


这是因为每个模块对象module都有一个属性:loaded。为false表示还没有加载,为true表示已经加载。


  • 如果有循环引入,那么加载顺序是什么?


网络异常,图片无法展示
|


如上图,Node采用的是深度优先算法:main -> aaa -> ccc -> ddd -> eee ->bbb


CommonJS规范缺点


  • CommonJS加载模块是同步的:


  • 同步的意味着只有等到对应的模块加载完毕,当前模块中的内容才能被运行。


  • 这个在服务器不会有什么问题,因为服务器加载的js文件都是本地文件,加载速度非常快。


如果将它应用于浏览器呢?


  • 浏览器加载js文件需要先从服务器将文件下载下来,之后再加载运行。


  • 那么采用同步的就意味着后续的js代码都无法正常运行,即使是一些简单的DOM操作。所以在浏览器中,我们通常不使用CommonJS规范。当然在webpack中使用CommonJS是另外一回事。因为它会将我们的代码转成浏览器可以直接执行的代码。


AMD规范


在早期为了可以在浏览器中使用模块化,通常会采用AMD或CMD。但是目前一方面现代的浏览器已经支持ES Modules,另一方面借助于webpack等工具可以实现对CommonJS或者ES Module代码的转换。AMD和CMD已经使用非常少了,所以这里我们进行简单的演练。


AMD主要是应用于浏览器的一种模块化规范:


  • AMD是Asynchronous Module Definition(异步模块定义)的缩写。它采用的是异步加载模块。


  • 我们提到过,规范只是定义代码的应该如何去编写,只有有了具体的实现才能被应用。


  • AMD实现的比较常用的库是require.js和curl.js。 require.js的使用



  • 定义HTML的script标签引入require.js和定义入口文件。data-main属性的作用是在加载完src的文件后会加载执行该文件


// index.html
 <script src="./require.js" data-main="./index.js"></script>


//main.js
    require.config({
      baseUrl: '', // 默认是main.js的文件夹路径
      paths: {
        foo: "./foo"
      }
    })
    require(["foo"], function(foo) {
      console.log("main:", foo)
    })


// foo.js
    define(function() {
      const name = "zh"
      const age = 22
      function sum(num1, num2) {
        return num1 + num2
      }
      return {
        name,
        age,
        sum
      }
    })


CMD规范


CMD规范也是应用于浏览器的一种模块化规范:


  • CMD 是Common Module Definition(通用模块定义)的缩写。它也采用了异步加载模块,但是它将CommonJS的优点吸收了过来。


  • CMD实现的比较常用的库是SeaJS。 SeaJS的使用



  • 引入sea.js和使用主入口文件。


// index.html
      <script src="./sea.js"></script>
      <script>
        seajs.use("./main.js")
      </script>


//main.js
    define(function(require, exports, module) {
      const foo = require("./foo")
      console.log("main:", foo)
    })


// foo.js
   define(function(require, exports, module) {
      const name = "zh"
      const age = 22
      function sum(num1, num2) {
        return num1 + num2
      }
      // exports.name = name
      // exports.age = age
      module.exports = {
        name,
        age,
        sum
      }
    });


ES Module


ES Module和CommonJS的模块化有一些不同之处:


  • 一方面它使用了import和export关键字来实现模块化。


  • 另一方面它采用编译期的静态分析,并且也加入了动态引用的方式。


  • export负责将模块内的内容导出。


  • import负责从其他模块导入内容。


  • 采用ES Module将自动采用严格模式:use strict。


基本使用


// index.html
    <script src="./main.js" type="module"></script>


// foo.js
    let obj = {
      name: "zh",
      age: 22
    }
    export default sum


// main.js
    import foo from './foo.js'
    console.log(foo) 


  • 在html文件加载入口文件的时候,需要指定type为module。


  • 在打开html文件时,需要开启本地服务,而不能直接打开运行在浏览器上。


网络异常,图片无法展示
|


这个在MDN上面有给出解释:


  • 你需要注意本地测试 — 如果你通过本地加载Html 文件 (比如一个 file:// 路径的文件), 你将会遇到 CORS 错误,因为Javascript 模块安全性需要。


  • 你需要通过一个服务器来测试。


exports关键字


export关键字将一个模块中的变量、函数、类等导出。


我们希望将其他中内容全部导出,它可以有如下的方式:


  • 方式一:在语句声明的前面直接加上export关键字。


export const name = "zh"
    export const age = 22


  • 方式二:将所有需要导出的标识符,放到export后面的 {} 中。注意:这里的 {}里面不是ES6的对象字面量的增强写法,{}也不是表示一个对象的。所以: export {name: name},是错误的写法。


const name = "zh"
    const age = 22
    function foo() {
      console.log("foo function")
    }
    export {
      name,
      age,
      foo
    }


  • 方式三:导出时给标识符起一个别名。(基本没用,一般在导入文件中起别名)。然后在导入文件中就只能使用别名来获取。


export {
      name as fName,
      age as fAge,
      foo as fFoo
    }


import关键字


import关键字负责从另外一个模块中导入内容。


导入内容的方式也有多种:


  • 方式一:import {标识符列表} from '模块'。注意:这里的{}也不是一个对象,里面只是存放导入的标识符列表内容。


import { name, age } from "./foo.js"


  • 方式二:导入时给标识符起别名。


import { name as fName, age as fAge } from './foo.js'


  • 方式三:通过 * 将模块功能放到一个模块功能对象(a module object)上。然后通过起别名来使用。


import * as foo from './foo.js'


export和import结合使用


表示导入导出。


import { add, sub } from './math.js'
    import {otherProperty} from './other.js'
    export {
      add,
      sub,
      otherProperty
    }


等价于


// 导入的所有文件会统一被导出
    export { add, sub } from './math.js'
    export {otherProperty} from './other.js'


等价于


export * from './math.js'
    export * from './other.js'


为什么要这样做呢?


在开发和封装一个功能库时,通常我们希望将暴露的所有接口放到一个文件中。 这样方便指定统一的接口规范,也方便阅读。这个时候,我们就可以使用export和import结合使用。


default用法


前面我们学习的导出功能都是有名字的导出(named exports):


  • 在导出export时指定了名字。


  • 在导入import时需要知道具体的名字。


还有一种导出叫做默认导出(default export)


// foo.js
    const name = "zh"
    cconst age = 22
    export {
      name,
      // 或者这样的默认导出
      // age as default
    }
    export default age


// 导入语句: 导入的默认的导出
    import foo, {name} from './foo.js'
    console.log(foo, name) // 22 zh


  • 默认导出export时可以不需要指定名字。


  • 在导入时不需要使用 {},并且可以自己来指定名字。


  • 它也方便我们和现有的CommonJS等规范相互操作。注意:在一个模块中,只能有一个默认导出(default export)。


import函数


通过import加载一个模块,是不可以在其放到逻辑代码中的,比如:


if(true) {
        import foo from './foo.js'
    }


为什么会出现这个情况呢?


  • 这是因为ES Module在被JS引擎解析时,就必须知道它的依赖关系。


  • 由于这个时候js代码没有任何的运行,所以无法在进行类似于if判断中根据代码的执行情况。


但是某些情况下,我们确确实实希望动态的来加载某一个模块:


  • 如果根据不同的条件,动态来选择加载模块的路径。 这个时候我们需要使用 import() 函数来动态加载。import函数返回的结果是一个Promise。


import("./foo.js").then(res => {
      console.log("res:", res.name)
    })


es11新增了一个属性。meta属性本身也是一个对象: { url: "当前模块所在的路径" }


console.log(import.meta)


ES Module的解析流程


ES Module是如何被浏览器解析并且让模块之间可以相互引用的呢?


详细请参考这篇文章


ES Module的解析过程可以划分为三个阶段:


  • 阶段一:构建(Construction),根据地址查找js文件,并且下载,将其解析成模块记录(Module Record)。


  • 阶段二:实例化(Instantiation),对模块记录进行实例化,并且分配内存空间,解析模块的导入和导出语句,把模块指向对应的内存地址。


  • 阶段三:运行(Evaluation),运行代码,计算值,并且将值填充到内存地址中。

网络异常,图片无法展示
|


阶段一:


网络异常,图片无法展示
|


阶段二和三:


网络异常,图片无法展示
|


所以,从上面可以看出在导出文件中,修改变量的值会影响到导入文件中的值。而且导入文件被限制修改导出文件的值。


es6 module 和common.js的区别


相关文章
|
2月前
|
JavaScript UED
js之模块化(2)
js之模块化(2)
|
3月前
|
JavaScript 前端开发 编译器
解锁JavaScript模块化编程新纪元:从CommonJS的基石到ES Modules的飞跃,探索代码组织的艺术与科学
【8月更文挑战第27天】随着Web应用复杂度的提升,JavaScript模块化编程变得至关重要,它能有效降低代码耦合度并提高项目可维护性及扩展性。从CommonJS到ES Modules,模块化标准经历了显著的发展。CommonJS最初专为服务器端设计,通过`require()`同步加载模块。而ES Modules作为官方标准,支持异步加载,更适合浏览器环境,并且能够进行静态分析以优化性能。这两种标准各有特色,但ES Modules凭借其更广泛的跨平台兼容性和现代语法逐渐成为主流。这一演进不仅标志着JavaScript模块化的成熟,也反映了整个JavaScript生态系统的不断完善。
52 3
|
3月前
|
存储 JavaScript 前端开发
Node.js的基本语法
【8月更文挑战第12天】Node.js的基本语法
128 1
|
27天前
|
JavaScript 前端开发
JavaScript 函数语法
JavaScript 函数是使用 `function` 关键词定义的代码块,可在调用时执行特定任务。函数可以无参或带参,参数用于传递值并在函数内部使用。函数调用可在事件触发时进行,如用户点击按钮。JavaScript 对大小写敏感,函数名和关键词必须严格匹配。示例中展示了如何通过不同参数调用函数以生成不同的输出。
|
2月前
|
JavaScript 前端开发 开发者
js之模块化(1)
js之模块化(1)
|
1月前
|
JavaScript 前端开发 大数据
在JavaScript中,Object.assign()方法或展开语法(...)来合并对象,Object.freeze()方法来冻结对象,防止对象被修改
在JavaScript中,Object.assign()方法或展开语法(...)来合并对象,Object.freeze()方法来冻结对象,防止对象被修改
16 0
|
1月前
|
缓存 JavaScript 前端开发
Node.js模块化的基本概念和分类及使用方法
Node.js模块化的基本概念和分类及使用方法
32 0
|
3月前
|
JavaScript 前端开发
JavaScript基础&实战(1)js的基本语法、标识符、数据类型
这篇文章是JavaScript基础与实战教程的第一部分,涵盖了JavaScript的基本语法、标识符、数据类型以及如何进行强制类型转换,通过代码示例介绍了JS的输出语句、编写位置和数据类型转换方法。
JavaScript基础&实战(1)js的基本语法、标识符、数据类型
|
2月前
|
JavaScript
js之模块化(3)
js之模块化(3)
|
3月前
|
前端开发 JavaScript 程序员
前端 JavaScript 的 _ 语法是个什么鬼?
前端 JavaScript 的 _ 语法是个什么鬼?