编译器是怎么工作的 —— 代码生成

前两篇博客已经把源代码分析和分析结果转换的这两个过程写完了,还剩下最后一个过程 —— 新代码的生成。

新代码的生成

codeGenerator.js

export default function codeGenerator(node) {
  switch (node.type) {
    case 'Program':
      return node.body.map(codeGenerator)
        .join('\n');
    case 'ExpressionStatement':
      return (
        codeGenerator(node.expression) +
        ';'
      );
    case 'CallExpression':
      return (
        codeGenerator(node.callee) +
        '(' +
        node.arguments.map(codeGenerator)
          .join(', ') +
        ')'
      );
    case 'Identifier':
      return node.name;
    case 'NumberLiteral':
      return node.value;
    case 'StringLiteral':
      return '"' + node.value + '"';
    default:
      throw new TypeError(node.type);
  }
}

test.js

import tokenizer from './tokenizer'
import parser from './parser'
import transformer from './transformer'
import codeGenerator from './codeGenerator'

const tokenizer_res = tokenizer('(add 2 (subtract 4 2))')
const parser_res = parser(tokenizer_res)
const transformer_res = transformer(parser_res)
const codeGenerator_res = codeGenerator(transformer_res)
console.log(codeGenerator_res)

// add(2, subtract(4, 2));

这一步就是将 transformer 输出的新 AST 处理并生成 JS 代码字符串的过程。

具体来说,switch 语句根据传给 codeGenerator 的节点对象(node)的 type 属性来决定采用哪种处理方式:

  • 如果 typeProgram,将节点对象的 body 属性进行 map,处理函数为 codeGenerator,实际上是将 body 中的元素一一递归,返回了数组后,元素间进行换行,但该例中 body 只有一个元素不会换行。最终将输出的字符串返回。

  • 如果 typeExpressionStatement,那么将节点对象的 expression 属性交给 codeGenerator 进行递归,并将返回的结果加上分号之后一并返回。

  • 如果 typeCallExpression,那么返回的字符串是这样的,将节点对象的 callee 属性交给 codeGenerator 递归,递归的结果拼上左括号,再拼上 node.arguments.map(codeGenerator).join(', ')(这类似于 typeProgram 时的处理方式,只是这时要 map 的数组是 arguments,产生的数组元素间加上逗号和空格),最后再拼上右括号,完成之后将结果返回。

  • 如果 typeIdentifier,返回节点对象的 name 属性。

  • 如果 typeNumberLiteral,返回节点对象的 value 属性。

  • 如果 typeStringLiteral,节点对象的 value 属性加上双引号之后返回,但该例中不存在 StringLiteral

  • 如果都不是,抛出一个错误。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML标准。 注意:讲述HT...
    kismetajun阅读 27,678评论 1 45
  • 最近一朋友去了趟美西,跟我讲了一个比较尴尬的事,她住的酒店没有热水壶,所以直接喝了水龙头的热水。。。。她朋友一脸严...
    至美旅行阅读 13,093评论 0 2
  • ▲形容词这些五颜六色终归要消褪荒诞夸张或讽刺的句子都如水漂般沉入湖底涟漪是唯一存在过的证据湖面破碎一个人在涌动他激...
    草夫阅读 297评论 1 0
  • 再多的怦然心动 也抗衡不了性格不对的互相折磨 抵不住天长地久的消耗 最终还是要和对的人在一起过 才不枉此生 就是...
    卢娅欣阅读 141评论 0 0
  • 以前写过很多博客,但都是给自己看的,是一些知识总结和生活感受。现在,要写一个给别人看的博客,希望读者能通过阅读获得...
    孙和David阅读 269评论 0 0