首页 > 其他分享 >@babel-AST常用方法

@babel-AST常用方法

时间:2023-05-06 13:22:25浏览次数:42  
标签:常用 code const AST babel path 节点 types

@babel常用方法

一、js对应AST

  1. obj变量名对应Identifier,常量串对应StringLiteral,数字对应NumericLiteral

  2. CallExpression主要关注calleearguments属性,分别表示被调用的函数和参数列表。

  3. MemberExpression主要关注objectpropertycomputed属性,分别表示对象,属性和是否是计算属性。Dot NotationArray Notationcomputed分别为falsetrue

二、@babel/parser方法

  • parser.parse(code, [{options}]):解析一段 JavaScript 代码;

  • parser.parseExpression(code, [{options}]):考虑到了性能问题,解析单个 JavaScript 表达式。

    部分可选参数 options

    参数描述
    allowImportExportEverywhere 默认 importexport 声明语句只能出现在程序的最顶层,设置为 true 则在任何地方都可以声明
    allowReturnOutsideFunction 默认如果在顶层中使用 return 语句会引起错误,设置为 true 就不会报错
    sourceType 默认为 script,当代码中含有 importexport 等关键字时会报错,需要指定为 module
    errorRecovery 默认如果 babel 发现一些不正常的代码就会抛出错误,设置为 true 则会在保存解析错误的同时继续解析代码,错误的记录将被保存在最终生成的 AST 的 errors 属性中,当然如果遇到严重的错误,依然会终止解析

三、@babel/generator方法

可以将 AST 还原成 JavaScript 代码,提供了一个 generate 方法:generate(ast, [{options}], code)

部分可选参数 options

参数描述
auxiliaryCommentBefore 在输出文件内容的头部添加注释块文字
auxiliaryCommentAfter 在输出文件内容的末尾添加注释块文字
comments 输出内容是否包含注释
compact 输出内容是否不添加空格,避免格式化
concise 输出内容是否减少空格使其更紧凑一些
minified 是否压缩输出代码
retainLines 尝试在输出代码中使用与源代码中相同的行号

四、@babel/traverse方法

通常和 visitor 一起使用,visitor 是一个对象,这个名字是可以随意取的,visitor 里可以定义一些方法来过滤节点

path 对象,该对象的类型是 NodePath,该对象有非常多的属性,以下介绍几种最常用的:

属性描述
toString() 当前路径的源码
node 当前路径的节点
parent 当前路径的父级节点
parentPath 当前路径的父级路径
type 当前路径的类型
  • replaceWith:用一个节点替换另一个节点;

  • replaceWithMultiple:用多个节点替换另一个节点;

  • replaceWithSourceString:将传入的源码字符串解析成对应 Node 后再替换,性能较差,不建议使用;

  • replaceInline:用一个或多个节点替换另一个节点,相当于同时有了前两个函数的功能。

五、@babel/types方法

主要用于构建新的 AST 节点或者节点类型判断

const parser = require("@babel/parser");
const generate = require("@babel/generator").default
const traverse = require("@babel/traverse").default
const types = require("@babel/types")

const code = `
const a = 1;
`
const ast = parser.parse(code)
const visitor = {
   //从VariableDeclaration这个节点开始遍历添加
   VariableDeclaration(path) {

       left = types.identifier("b")
       right = types.identifier("a")
       //将左右节点用+符号拼接
       let BinaryExpression = types.binaryExpression("+", left, right)
       //添加到return节点
       let returnStatement = types.returnStatement(BinaryExpression)
       //添加到作用域{}节点
       let BlockStatement = types.blockStatement([returnStatement])
       //添加函数节点,函数名为func,参数为a,b
       declaration = types.functionDeclaration(types.identifier("func"), [types.identifier("a"),types.identifier("b")], BlockStatement)
       path.insertAfter(declaration)

       //将a节点和数字节点用*符号拼接
       left = types.binaryExpression("*", types.identifier("a"), types.numericLiteral(5))
       right = types.numericLiteral(1)
       //将左右节点用+符号拼接
       let value = types.binaryExpression("+", left, right)
       //创建一个变量b,将value赋值于b
       let declarator = types.variableDeclarator(types.identifier("b"), value)
       declaration = types.variableDeclaration("const", [declarator])
       path.insertAfter(declaration) //插入节点

       path.stop() //停止遍历节点
  },

}

traverse(ast, visitor)
const result = generate(ast)
console.log(result.code)
const parser = require("@babel/parser");
const generate = require("@babel/generator").default
const traverse = require("@babel/traverse").default
const types = require('@babel/types');

const code = `
const example = function () {
   let a;
   if (false) {
       a = 1;
   } else {
       if (1) {
           a = 2;
       }
       else {
           a = 3;
       }
   }
   return a;
};
`
const ast = parser.parse(code)

const visitor = {
   enter(path) {
       if (types.isBooleanLiteral(path.node.test) || types.isNumericLiteral(path.node.test)) {
           //判断一定为真,就替换
           //if 对应的是 consequent 节点,else 对应的是 alternate 节点
           if (path.node.test.value) {
               path.replaceInline(path.node.consequent.body); //方法将节点替换成计算结果生成的新节点
          } else {
               if (path.node.alternate) {
                   path.replaceInline(path.node.alternate.body);
              } else {
                   path.remove()
              }
          }
      }
  }
}

traverse(ast, visitor)
const result = generate(ast)
console.log(result.code)
 

标签:常用,code,const,AST,babel,path,节点,types
From: https://www.cnblogs.com/angelyan/p/17376971.html

相关文章

  • Elasticsearch模糊查询
    Elasticsearch模糊查询ES模糊查询共有三种:1.match分词匹配查询和match_phrase短语查询Elasticsearch的match_phrase短语查询跟match的区别就是,关键词作为一个整体进行搜索,而不是拆分成一个个关键词。当匹配对象为中文时,建议使用:MatchPhraseQueryBuilderquery=......
  • Python之路,Day21 - 常用算法学习
    本节内容算法定义时间复杂度空间复杂度常用算法实例 1.算法定义算法(Algorithm)是指解题方案的准确而完整的描述,是一系列解决问题的清晰指令,算法代表着用系统的方法描述解决问题的策略机制。也就是说,能够对一定规范的输入,在有限时间内获得所要求的输出。如果一个算法有缺陷,......
  • PostgresQL-toast表数据损坏修复案例
    PostgresQL-toast表数据损坏修复案例pg_statistic-toast表块损坏修复当访问该表的时候报错:missingchunknumber0fortoastvalue59747165inpg_toast_2619首先我们要找到出问题的表:toast表的表名是字符串"pg_toast"与表tbl的oid"2619"拼接而成,可以直接查询oid=26191、......
  • 解密Elasticsearch:深入探究这款搜索和分析引擎
    作者:京东保险 管顺利开篇最近使用Elasticsearch实现画像系统,实现的dmp的数据中台能力。同时调研了竞品的架构选型。以及重温了redis原理等。特此做一次es的总结和回顾。网上没看到有人用Elasticsearch来完成画像的。我来做第一次尝试。背景说完,我们先思考一件事,使用内存系统做......
  • 上传代码到gitee时使用了git pull --rebase origin master命令导致代码丢失
    转自:https://blog.csdn.net/m0_51557131/article/details/129844800 上传本地代码到码云仓库步骤:进入项目目录gitinit将本地项目工作区的所有文件添加到暂存区   gitadd xxx #xxx代表项目文件名字,及所要提交的内容文件   gitadd.  #.表示要提交当前地址下的......
  • Linux安装rabbitMQ常用命令
    1.拉取最新的rabbitMQdockerpullrabbitmq:management2.容器启动rabbitMQdockerrun-d--hostnamemy-rabbit--namerabbit-p15672:15672-p5672:5672rabbitmq:management其中:     --hostname:指定容器主机名称     --name:        指定容器名称  ......
  • SSH客户端常用工具SecureCRT操作
    1.1SecureCRT工具介绍SecureCRT是一款支持SSH(SSH1和SSH2)协议的终端仿真软件,常被用来运行于Windows下远程登录UNIX或Linux服务器。SecureCRT软件功能强大,不仅仅支持SSH协议,同时还支持Telnet、RLogin、Serial和TAPI等协议,它有非常多的功能,这里就不一一介绍了,常用功能可见下文介绍......
  • 常用的截取字符串方法JS和Golang实现
    JS中截取字符串很简单,直接使用substr函数substr()方法可在字符串中截取从开始下标开始的指定数目的字符。下标是从0开始算例如:"21".substr(0,1)  返回2golang实现的substr//截取字符串,支持多字节字符//start:起始下标,负数从从尾部开始,最后一个为-1//length:截取长度,......
  • maven常用命令
    初学maven即感受到其强大的便捷性,大体记录下maven的基础知识以备不时之需。常用命令为:   mvnarchetype:create:创建Maven项目   mvncompile:编译源代码   mvntest-compile:编译测试代码   mvntest:运行应用程序中的单元测试   mvnsite:生成项目相关......
  • swagger3 常用注解
    swagger2OpenAPI3注解位置@Api@Tag(name=“接口类描述”)Controller类上@ApiOperation@Operation(summary=“接口方法描述”)Controller方法上@ApiImplicitParams@ParametersController方法上@ApiImplicitParam@Parameter(description=“参数描......