新建一个项目,再新建一个src文件夹,里面有三个文件,word.js,message.js,index.js

word.js

export const word = 'hello';

message.js

import { word } from './word.js';
const message = `say ${word}`;
export default message;

index.js

import message from './message.js';
console.log(message);

这三个模块通过调用,最后打印出say hello。那我们说,如果直接想src目录下的代码运行下浏览器下,是不可以的;浏览器根本就不认识这种语法。所以我们需要通过webpack类似的打包工具帮助我们进行项目的打包。在根目录下创建一个bundler.js,这就是我们要做的打包工具

首先打包工具肯定是用node.js来做开发的,所以需要安装node。用了webpack,肯定是装了node的。
bundler.js
const fs = require('fs'); // 帮助我们获取一些文件的信息

// 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  console.log(content);
} moduleAnalyser('./src/index.js');

运行node bundler.js,输出出来的就是index.js里面的内容。这里在控制台显示的内容,黑色的文本,不是很好看。我们可以安装一个工具

sudo npm install cli-highlight -g
然后运行 node bundler.js | highlight
这个时候,打印出来的代码就会高亮。
拿到文件内容以后。需要分析这个文件里面有哪些依赖,怎么拿到依赖呢?字符串截取可以,但如果很多个模块引入的时候,字符串截取就会显得非常麻烦。我们安装babel的第三方模块帮助我们去分析依赖,安装
npm install @babel/parser --save
安装好之后我们就可以使用了
bundler.js
const fs = require('fs'); // 帮助我们获取一些文件的信息
const parser = require('@babel/parser'); // 帮助我们分析代码,引入的文件 // 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  console.log(parser.parse(content, {
    sourceType: 'module' // 说明是es module的引入方式
  }));
} moduleAnalyser('./src/index.js');
这个时候打印出来一个对象。这个对象就是我们听说的一个概念,叫做抽象语法树。也就是ast

这个对象可以很好的表述我当前的这段代码。这个对象里有一个program这样的一个字段,表示当前运行的程序,里面有个body字段,我们可以打印这个字段看一下。

bundler.js
const fs = require('fs'); // 帮助我们获取一些文件的信息
const parser = require('@babel/parser'); // 帮助我们分析代码,引入的文件 // 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  const ast = parser.parse(content, {
    sourceType: 'module' // 说明是es module的引入方式
  })
  console.log(ast.program.body);
} moduleAnalyser('./src/index.js');

这个时候打印出来body的两个节点,首先第一个节点ImportDeclaration,确实是import一些东西。也就是引入的声明。第二个节点是ExpressionStatement,是一个表达式的声明,console.log确实是一个表达式的语句。所以通过babel.parser可以分析出抽象语法树。通过抽象语法树,我们就可以找到一些声明的语句,而声明的语句,放置的就是入口文件对应的一些依赖关系。我们可以遍历body,找到ImportDeclaration这样的一些内容,如果自己写一些遍历的话,还是有点麻烦,babel还提供给我们一个模块,可以帮助我们快速的找到import的节点。所以我们需要安装这样的一个模块

npm install @babel/traverse --save

bundler.js

const fs = require('fs'); // 帮助我们获取一些文件的信息
const parser = require('@babel/parser'); // 帮助我们分析代码,引入的文件
const traverse = require('@babel/traverse').default;// 因为是export出来的内容,必须加一个default属性才可以 // 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  // 利用parser.parse获取到ast
  const ast = parser.parse(content, {
    sourceType: 'module' // 说明是es module的引入方式
  });
  // 利用traverse对代码进行一个分析
  const dependencies = [];
  traverse(ast, {
    // 只要抽象语法树有ImportDeclaration就会进入这个方法,node是节点
    ImportDeclaration({ node }){
      dependencies.push(node.source.value);
    }
  });
  console.log(dependencies);
} moduleAnalyser('./src/index.js');

再次运行 node bundler.js | highlight。发现已经打印出了依赖的模块。[ './message.js' ]。

这个时候对入口文件对依赖分析就分析出来了。但是这个路径是相对路径,相对于传进来对index.js。在真正做代码打包的时候,我们这些依赖文件他不能是相对路径。他必须是一个绝对路径。或者如果一定要相对路径,也要相对于bundler这个项目,打包才不会有问题。解决这个问题,引入node的另外一个核心模块,叫做path
const fs = require('fs'); // 帮助我们获取一些文件的信息
const path = require('path'); // 打包的时候需要绝对路径,借助path这个模块
const parser = require('@babel/parser'); // 帮助我们分析代码,引入的文件
const traverse = require('@babel/traverse').default;// 因为是export出来的内容,必须加一个default属性才可以 // 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  // 利用parser.parse获取到ast
  const ast = parser.parse(content, {
    sourceType: 'module' // 说明是es module的引入方式
  });
  // 利用traverse对代码进行一个分析
  const dependencies = [];
  traverse(ast, {
    // 只要抽象语法树有ImportDeclaration就会进入这个方法,node是节点
    ImportDeclaration({ node }){
      // 拿到filename对应的文件夹路径
      const dirname = path.dirname(filename);
      // 对这个文件夹的路径进行一个转化,将引入的模块转化成绝对路径
      const newFile = path.join(dirname, node.source.value);
      console.log(newFile);
      dependencies.push(node.source.value);
    }
  });
} moduleAnalyser('./src/index.js');

这个时候打印出来的就是引入的模块的绝对路径。入口文件和相对应的依赖,都可以分析出来了。但是我们的代码,浏览器还是不支持的,需要借助babel去解析我们es6的代码。

安装

npm install @babel/core --save
这是babel的核心模块
 
bundler.js
const fs = require('fs'); // 帮助我们获取一些文件的信息
const path = require('path'); // 打包的时候需要绝对路径,借助path这个模块
const parser = require('@babel/parser'); // 帮助我们分析代码,引入的文件
const traverse = require('@babel/traverse').default;// 因为是export出来的内容,必须加一个default属性才可以
const babel = require('@babel/core'); // babel的核心模块,转化代码,转化成浏览器认识的代码 // 分析模块
const moduleAnalyser = (filename) => {
  // 读取文件内容
  const content = fs.readFileSync(filename, 'utf-8');
  // 利用parser.parse获取到ast
  const ast = parser.parse(content, {
    sourceType: 'module' // 说明是es module的引入方式
  });
  // 利用traverse对代码进行一个分析
  const dependencies = {};
  traverse(ast, {
    // 只要抽象语法树有ImportDeclaration就会进入这个方法,node是节点
    ImportDeclaration({ node }){
      // 拿到filename对应的文件夹路径
      const dirname = path.dirname(filename);
      // 对这个文件夹的路径进行一个转化,将引入的模块转化成相对于bundler的相对路径
      const newFile = './' + path.join(dirname, node.source.value);
      // 为了方便,把相对路径,绝对路径都存上,key是相对路径,value是绝对路径
      dependencies[node.source.value] = newFile;
    }
  });
  // 这个方法可以将抽象语法树转化成浏览器可以运行代码。
  const { code } = babel.transformFromAst(ast, null, {
    presets: ['@babel/preset-env'] // 把es6语法翻译成es5语法
  });
  // 返回入口文件和相对应的依赖,都可以分析出来了。
  return {
    filename,
    dependencies,
    code
  }
} const moduleInfo = moduleAnalyser('./src/index.js');
console.log(moduleInfo);

这个时候就分析好了文件该有的内容,入口文件,对应的依赖,翻译好的代码。那么接下来就是分析其他的文件

webpack中bundler源码编写的更多相关文章

  1. webpack中bundler源码编写2

    通过第一部分的学习,我们已经可以分析一个js的文件.这节课我们学习Dependencies Graph,也就是依赖图谱.对所有模块进行分析.先分析index.js.index.js里面引入了messg ...

  2. php中foreach源码分析(编译原理)

    php中foreach源码分析(编译原理) 一.总结 编译原理(lex and yacc)的知识 二.php中foreach源码分析 foreach是PHP中很常用的一个用作数组循环的控制语句.因为它 ...

  3. vue打包时,assets目录 和static目录下文件的处理区别(nodeModule中插件源码修改后,打包后的文件应放在static目录)

    为了回答这个问题,我们首先需要了解Webpack如何处理静态资产.在 *.vue 组件中,所有模板和CSS都会被 vue-html-loader 及 css-loader 解析,并查找资源URL.例如 ...

  4. 【原】Spark中Client源码分析(二)

    继续前一篇的内容.前一篇内容为: Spark中Client源码分析(一)http://www.cnblogs.com/yourarebest/p/5313006.html DriverClient中的 ...

  5. 【原】Spark中Master源码分析(二)

    继续上一篇的内容.上一篇的内容为: Spark中Master源码分析(一) http://www.cnblogs.com/yourarebest/p/5312965.html 4.receive方法, ...

  6. 【原】 Spark中Worker源码分析(二)

    继续前一篇的内容.前一篇内容为: Spark中Worker源码分析(一)http://www.cnblogs.com/yourarebest/p/5300202.html 4.receive方法, r ...

  7. Django缓存机制--rest_framework中节流源码使用的就是django提供的缓存api

    一.配置缓存   https://www.jb51.net/article/124434.htm 二.缓存全站.页面.局部   三.自我控制的简单缓存API API 接口为:django.core.c ...

  8. 深入理解 Node.js 中 EventEmitter源码分析(3.0.0版本)

    events模块对外提供了一个 EventEmitter 对象,即:events.EventEmitter. EventEmitter 是NodeJS的核心模块events中的类,用于对NodeJS中 ...

  9. 从 sourcemap 中获取源码

    使用 paazmaya/shuji: Reverse engineering JavaScript and CSS sources from sourcemaps 可以从 sourcemap 中获取源 ...

随机推荐

  1. LODOP中的各种宽高和位置简短问答

    LODOP中的位置边距,可查看本博客另一篇相关博文:LODOOP中的各种边距 打印项.整体偏移.可打区域.内部边距关于LODOP中的各种宽高,可查看本博文简短问答下方的正文:.该文其实有两个以前的相关 ...

  2. jQuery调用WebService返回JSON数据

    相信大家都比较了解JSON格式的数据对于ajax的方便,不了解的可以从网上找一下这方面的资料来看一下,这里就不多说了,不清楚的可以在网上查一下,这里只说一下因为参数设置不当引起的取不到返回值的问题. ...

  3. 支持“ReportDbContext”上下文的模型已在数据库创建后发生更改

    支持“ReportDbContext”上下文的模型已在数据库创建后发生更改.请考虑使用 Code First 迁移更新数据库(http://go.microsoft.com/fwlink/?LinkI ...

  4. Spring MVC如何获取请求中的参数

    目录 一.获取URL中路径参数 1.1 @PathVariable 注解 1.2 @PathParam 注解 二.获取请求参数: 2.1 GET请求 2.1.1 获取请求中的单个参数:@Request ...

  5. QuickTime专业版 pro 注册码

    打开QuickTime Player下拉编辑菜单--选偏好设置--注册 Name: Dawn M Fredette Key: 4UJ2-5NLF-HFFA-9JW3-X2KV 重新启动 QuickTi ...

  6. Linux查看库依赖方法

    1.查看依赖的库:objdump -x xxx.so | grep NEEDED 2.查看可执行程序依赖的库:objdump -x 可执行程序名 | grep NEEDED 3.查看缺少的库: ldd ...

  7. input 标签鼠标放入输入框补全提示

    JSP: <input type="text" placeholder="eventDesc" value="" id="e ...

  8. lnmp+tp5安装纪要

    1: lnmp : https://lnmp.org/install.html  官网安装帮助 运行命令:wget http://soft.vpser.net/lnmp/lnmp1.6.tar.gz ...

  9. IntelliJ IDEA - 查找代码提交人

    转载. https://blog.csdn.net/abcyyjjkk/article/details/88995503 如果Annocation不可用

  10. python 之 数据库(视图、触发器、事务、存储过程)

    10.13 视图 1.什么是视图 视图就是通过查询得到一张虚拟表,然后保存下来,下次用的直接使用即可 2.为什么要用视图 如果要频繁使用一张虚拟表,可以不用重复查询 3.如何使用视图 视图记录的增.删 ...