待入门的SWC

SWC是一个基于Rust的JS编译器,旨在替代Babel,提供更快的转译性能。它包含了CLI、loader和核心API,用于与webpack和其它构建工具集成。尽管在某些场景下swc-loader的性能不如babel-loader,但其核心API提供了直接转换AST的能力,且允许开发者创建自定义插件。未来,如何优化配置和开发插件将是值得关注的方向。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

cf46207ca03ce6ffbf1e1f42e1e709ce.jpeg

前情提要

1. rust 是什么

  1. 一种编程语言,一度被赋予是c++/c语言的替代品

  2. 支持函数式,面向对象编程

2. 编译器的基本工作流程

  1. 三个阶段:解析、转换、代码生成

  • 解析:将原始代码(字符) -> 词法分析 -> 令牌 -> 语法分析 -> 抽象语法树(AST)

  • 转换:处理抽象语法树(AST)-> 转换器 -> 新的语法树(AST)

  • 代码生成:将处理后的新的语法树(AST)->  新的代码(字符)

9b13b96eda59e6cdecfc7d175a1af8c8.png

SWC为什么会诞生

既生瑜何生亮

swc被推崇,除了大众追捧外,也一定程度上说明babel 存在一些不太容易提升的地方。引用社区所言:

  1. 语言劣势,用JS写的babel 不能用多核cpu处理编译任务链 

1ea9a1f2916bb0c03f54ff431ac8525c.png

“这其中转换为AST以及编译成字节码应该是最耗费性能的”

SWC是什么

  • Speedy Web Compiler

  • 基于Rust语言的JS编译器(Javascript Compiler),其生态周边中包含压缩插件、打包工具spack

  • 主要对标Babel,誓言要代替Babel (据说:转译性能比Babel快20倍)

相关应用形式
  1. @swc/cli: 自带了一个内置的 CLI 命令行工具,可通过命令行编译文件,类似于@babel/cli

  2. swc-loader: 该模块使得可以与 webpack 一起使用,类似于babel-loader

  3. @swc/core:  核心API的集合,类似于@babel/core

  4. ......

接下来简单说一下:

@swc/cli 通过命令行调用

// 随意找一个工程项目

npm i @swc/cli @swc/core

time npx swc[babel] xx.tsx -o after.js
8984e1688a47f80f3aea8d7ac09e064b.png

确实会快,当然我的文件里需要转义的代码不太多。

swc-loader

更换项目配置里的babel-loader (@vue/cli-plugin-babel)

328c3173bac7aaeea2f13423549fa1d7.png

build后的耗时如下,由于没有设置swc配套的插件和预设,结果却是 swc-loader 更耗时。

d4c5744809e8973e2d38beac93dbf554.png 19c8d0e5a3299509a1eb8e5050b680db.png

当没有额外配置swc相关转换规则,复用babel相关的plugin和preset,结果却出乎意料,使用babel-loder是1.75s,swc-loader是3.55s。

@swc/core

在node或者@swc/cli的任务流中可以使用api的形式调用其提供的方法,一般在构建工具中使用。

// const babel = require('@babel/core')
const swc = require('@swc/core')

module.exports = (api, options) => {
    // babel.loadPartialConfigSync({ filename: api.resolve('src/main.js') })
    swc.loadPartialConfigSync({ filename: api.resolve('src/main.js') })
    
    const res = await swc.transform('xxx.js', {
      filename: "out.js",
      jsc:{
        parser: {
          target: 'es5'
        }
      }
    })
}
核心API

1. swc_ecma_parser 获得AST结构

**Rust版本**
 // 声明swc文件对象
 let fm = cm.new_source_file(
    FileName::Custom("test.js".into()), // 文件名
    "function foo() { console.log('foo')}".into(), // 文件里具体的代码
 );
 
// 声明 词法分析Lexer解析规则 
let lexer = Lexer::new(
    Syntax::Es(Default::default()),
    EsVersion::Es2016,
    StringInput::from(&*fm), // fm 里的代码
    None,
);

// 声明Parser对象
let capturing = Capturing::new(lexer);
let mut parser = Parser::new_from(capturing);

// 在JS中每个文件一般是一个Module 
let mut module = parser.parse_module();



**JS版本**
async parse(src: string, options?: ParseOptions, filename?: string): Promise<Program> {
    options = options || { syntax: "ecmascript" };
    options.syntax = options.syntax || "ecmascript";
    
    if (bindings) {
      const res = await bindings.parse(src, toBuffer(options), filename);
      return JSON.parse(res);
    }

一般得到的ast树形结构如下,与babel类似:

5ca90e29bc26f904e4153fe7d406c9f9.png e664dd77fab026eda98f8b9b2527b1ff.png

2. swc_ecma_transform 转换AST

838d09ec4548a50066f7e2f698c4027b.png
Rust版本
 // 遍历ast body体
 for item in module.body {
    // 当前的引用值与item匹配时,把item ref赋值给 var
    if let ModuleItem::ModuleDecl(ModuleDecl::Import(var)) = item {
        let source = &*var.src.value;
        if source == "antd" {
            for specifier in &var.specifiers {
                match specifier {
                    ImportSpecifier::Named(ref s) => {
                        let ident = format!("{}", s.local.sym);
                        specifiers.push(format!("antd/es/{}/style/index.css", ident.to_lowercase()));
                    }
                    ImportSpecifier::Default(ref s) => {}
                    ImportSpecifier::Namespace(ref ns) => {}
                }
            }
         }
    }
 
 
 JS版本
 transformSync(src, options) {
   const { plugin } = options, newOptions = __rest(options, ["plugin"]);
   // 是否有plugin
    if (plugin) {
        const m = typeof src === "string" ? this.parseSync(src, (_c = options === null || options === void 0 ? void 0 : options.jsc) === null || _c === void 0 ? void 0 : _c.parser, options.filename) : src;
        return this.transformSync(plugin(m), newOptions);
    }
   return bindings.transformSync(isModule ? JSON.stringify(src) : src, isModule, toBuffer(newOptions));
 }
 
 function loadBinding(dirname, filename = 'index', packageName) {  
    // 获取系统信息
    const triples = triples_1.platformArchTriples[PlatformName][ArchName];
    
    // 遍历信息
    for (const triple of triples) {
        if (packageName) {
            try {
                // 获取到需要加载的二进制文件路径
                // /Users/xx/swc-demo/node_modules/@swc/core-darwin-x64/swc.darwin-x64.node
                return require(
                  require.resolve(
                    `${packageName}-${triple.platformArchABI}`,
                    { paths: [dirname] }
                  ));
            }
        }
    }
}

依旧用上面的例子尝试一下:

6e8b3e1cf611ee3c67a47d87a685be2e.png

可以得到以下结果:

"[\n    1,\n    2,\n    3\n].map(function(i) {\n    return i + 1;\n});\n"

这里没有像我们理解的编译器中的三阶段,就直接可以得到新的代码段。

与babel相似点:

  1. 在traverse转换ast过程中,都会基于helpers、plugins/preset 的规则进行转换

  2. plugins和preset的执行顺序一致

与babel 的不同点:

  1. 没有类似 @babel/generator 生成新代码,transform 阶段就可以生成。

// 声明compiler对象
let compiler = Compiler::new(fm.clone());

//生成新的ast
let mut newmodule = module.clone() as Module;

//调用Compile对象的print方法生成新的代码
let new_res = compiler.print(&newmodule,
EsVersion::Es2016,
...args).unwrap();
  1. visitor对象中没有enter/exit 钩子,由顶层的 visitProgram 往内递归执行节点访问操作。(看源码中被注释掉了,可能未来会支持)

面向js 生态的插件系统

与babel类似,swc/core也暴露一些api给开发者,可以自定义转换代码的插件。但官网提到目前有些性能问题。

可能的后续

  1. 如何在项目中合理化配置swc

  2. 如何开发一个swc插件

想了解更多转转公司的业务实践,点击关注下方的公众号吧!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值