React第五章(swc)

swc

什么是swc?

在这里插入图片描述

SWC 既可用于编译,也可用于打包。对于编译,它使用现代 JavaScript 功能获取 JavaScript / TypeScript 文件并输出所有主流浏览器支持的有效代码。

SWC在单线程上比 Babel 快 20 倍,在四核上快 70 倍。

简单点来说swc实现了和babel一样的功能,但是它比babel快。

FAQ为什么快?

  1. 编译型
    Rust 是一种编译型语言,在编译时将代码转化为机器码(底层的 CPU 指令)。这种机器码在执行时非常高效,几乎不需要额外的开销。

  2. 解释型
    JavaScript 是一种解释型语言,通常在浏览器或 Node.js 环境中通过解释器运行。尽管现代的 JavaScript 引擎(如 V8 引擎)使用了 JIT(即时编译)技术来提高性能,但解释型语言本质上还是需要更多的运行时开销。

v8编译原理 https://juejin.cn/post/7291135064843304994#heading-0

swc官网 https://swc.rs/

核心功能

  1. JavaScript/TypeScript 转换 可以将现代 JavaScript(ES6+)和 TypeScript 代码转换为兼容旧版 JavaScript 环境的代码。这包括语法转换(如箭头函数、解构赋值等)以及一些 polyfill 的处理
  2. 模块打包 SWC 提供了基础的打包功能,可以将多个模块捆绑成一个单独的文件
  3. SWC 支持代码压缩和优化功能,类似于 Terser。它可以对 JavaScript 代码进行压缩,去除不必要的空白、注释,并对代码进行优化以减小文件大小,提高加载速度
  4. SWC 原生支持 TypeScript,可以将 TypeScript 编译为 JavaScript
  5. SWC 支持 React 和 JSX 语法,可以将 JSX 转换为标准的 JavaScript 代码。它还支持一些现代的 React 特性

案例

1. 语法转换:将新版本的 JavaScript 语法转换为旧版本的语法

转换前

//语法
const a = (params = 2) => 1 + params;
const b = [1, 2, 3]
const c = [...b, 4, 5]
class Babel {

}
new Babel()
//API
const x = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10].filter((x) => x % 2 === 0)
const y = Object.assign({}, { name: 1 })

swc 转换代码

import swc from '@swc/core'

const result = swc.transformFileSync('./test.js', {
   jsc: {
       target: "es5", //代码转换es5
       parser: {
           syntax: 'ecmascript'
       }
   }
})
console.log(result.code)

转换后结果

//语法
function _array_like_to_array(arr, len) {
   if (len == null || len > arr.length) len = arr.length;
   for(var i = 0, arr2 = new Array(len); i < len; i++)arr2[i] = arr[i];
   return arr2;
}
function _array_without_holes(arr) {
   if (Array.isArray(arr)) return _array_like_to_array(arr);
}
function _class_call_check(instance, Constructor) {
   if (!(instance instanceof Constructor)) {
       throw new TypeError("Cannot call a class as a function");
   }
}
function _iterable_to_array(iter) {
   if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter);
}
function _non_iterable_spread() {
   throw new TypeError("Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
function _to_consumable_array(arr) {
   return _array_without_holes(arr) || _iterable_to_array(arr) || _unsupported_iterable_to_array(arr) || _non_iterable_spread();
}
function _unsupported_iterable_to_array(o, minLen) {
   if (!o) return;
   if (typeof o === "string") return _array_like_to_array(o, minLen);
   var n = Object.prototype.toString.call(o).slice(8, -1);
   if (n === "Object" && o.constructor) n = o.constructor.name;
   if (n === "Map" || n === "Set") return Array.from(n);
   if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _array_like_to_array(o, minLen);
}
var a = function() {
   var params = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : 2;
   return 1 + params;
};
var b = [
   1,
   2,
   3
];
var c = _to_consumable_array(b).concat([
   4,
   5
]);
var Babel = function Babel() {
   "use strict";
   _class_call_check(this, Babel);
};
new Babel();
//API
var x = [
   1,
   2,
   3,
   4,
   5,
   6,
   7,
   8,
   9,
   10
].filter(function(x) {
   return x % 2 === 0;
});
var y = Object.assign({}, {
   name: 1
});

swc转换用时 default: 8.088ms

Babel转换用时 default: 417.59ms

2. swc转换react jsx语法

test.jsx

import react from 'react'
import { createRoot } from 'react-dom/client'

const App = () => {
    return <div>小满是谁?????</div>
}

createRoot(document.getElementById('root')).render(<App />)

转换代码

import swc from '@swc/core'
console.time()
const result = swc.transformFileSync('./test.jsx', {
    jsc: {
        target: "es5", //代码转换es5
        parser: {
            syntax: 'ecmascript',
            jsx: true
        },
        transform:{
            react: {
                runtime: 'automatic'
            }
        }
    }
})
console.log(result.code)
console.timeEnd()

结果

import { jsx as _jsx } from "react/jsx-runtime";
import react from 'react';
import { createRoot } from 'react-dom/client';
var App = function() {
    return /*#__PURE__*/ _jsx("div", {
        children: "小满是谁?????"
    });
};
createRoot(document.getElementById('root')).render(/*#__PURE__*/ _jsx(App, {}));

swc转换用时 default: 4.251ms

Babel转换用时 default: 80.613ms

3.swc简易打包

截止 2024-9-4日 目前该功能鸡肋 不推荐使用 了解即可

目前swc打包只能支持cjs 未来才能支持esm 比较鸡肋 其次就是参数只能entry output暂无其他参数

创建配置文件spack.config.js

编写以下代码执行 npx spack打包

const { config } = require('@swc/core/spack')
const path = require('path')
module.exports = config({
    entry: {
        web: path.join(__dirname, './test.js') //入口
    },
    output: {
        path: path.join(__dirname, './dist'), //出口
        name: 'test.js'
    }
})
<think>好的,用户问的是“打包速度太慢如何优化”。首先,我需要确定用户的使用场景。通常打包速度慢的问题常见于前端开发,尤其是使用Webpack、Vite这样的构建工具的时候。可能用户是开发者或者项目维护者,遇到了构建时间过长,影响开发效率的问题。 接下来,我要考虑用户可能的具体情况。比如,项目规模较大,依赖较多,或者配置不当。用户可能已经尝试过一些基本的优化方法,但效果不明显,所以需要更深入的解决方案。 然后,我需要回忆常见的打包速度优化策略。这包括但不限于:升级依赖版本,使用更快的构建工具(如Vite替代Webpack),配置缓存,减少构建范围(比如通过exclude排除node_modules),代码分割,并行处理,使用DLL或者模块联邦,以及硬件层面的优化如SSD和更多内存。 还要考虑用户可能不太了解的高级技巧,比如利用持久化缓存、缩小文件检索范围、优化Loader和Plugin的使用,以及分析工具的使用(如Webpack Bundle Analyzer)。此外,区分开发和生产环境的不同配置也很重要,因为开发环境需要更快的构建,而生产环境可能需要更优的压缩。 同时,需要注意用户可能的误区,比如误用source map导致速度下降,或者未正确配置缓存。还要提醒用户权衡优化,比如压缩率和构建速度之间的平衡,避免过度优化影响其他方面。 最后,将这些思路组织成结构化的回答,分点说明,让用户能够逐步排查和优化。确保语言简洁明了,技术点准确,并提供示例配置帮助用户理解如何具体操作。同时,根据用户提供的上下文,之前讨论过1M以上数据压缩的问题,可能在这次回答中可以适当关联,比如提到压缩算法对构建速度的影响。</think>以下是针对 **打包速度优化** 的体系化解决方案,按优先级分步实施: --- ### 一、构建工具基准测试(关键第一步) ```bash # Webpack速度分析 npx webpack --profile --json > stats.json webpack-bundle-analyzer stats.json # Vite性能检测 vite build --profile vite visualize ``` **核心指标**: 1. 模块解析时间(Loader处理耗时) 2. 代码生成阶段占比 3. 插件调用次数与耗时 --- ### 二、通用加速策略(立即见效) ```mermaid graph LR A[打包慢] --> B{诊断方向} B --> C[模块解析] B --> D[代码生成] B --> E[产物优化] C --> C1[缩小文件搜索范围] C --> C2[避免重复编译] D --> D1[禁用SourceMap] D --> D2[关闭类型检查] E --> E1[代码分割优化] E --> E2[压缩算法选型] ``` #### 1. 路径解析优化(Webpack示例) ```javascript // webpack.config.js module.exports = { resolve: { extensions: ['.js', '.jsx'], // 减少扩展名尝试 modules: [path.resolve(__dirname, 'src'), 'node_modules'], // 限定搜索路径 alias: { '@': path.resolve(__dirname, 'src') // 路径别名 } }, module: { rules: [{ test: /\.js$/, exclude: /node_modules/, // 必须排除 use: ['babel-loader?cacheDirectory=true'] // 启用缓存 }] } } ``` #### 2. 并行处理加速 ```javascript const TerserPlugin = require('terser-webpack-plugin'); const HardSourceWebpackPlugin = require('hard-source-webpack-plugin'); module.exports = { cache: true, // 开启持久化缓存 parallelism: 4, // 控制并发数=CPU核心数×0.8 optimization: { minimizer: [ new TerserPlugin({ parallel: true, // 启用多进程压缩 cache: true, terserOptions: { compress: { drop_console: false // 保留console可提升10%压缩速度 } } }) ] }, plugins: [ new HardSourceWebpackPlugin() // 二级缓存 ] } ``` --- ### 三、进阶优化方案(需架构调整) #### 1. 模块联邦(Webpack 5+) ```javascript // host-app配置 new ModuleFederationPlugin({ name: 'host', remotes: { libs: 'libs@http://cdn.com/libs-remote-entry.js' }, shared: ['react', 'react-dom'] // 共享公共库 }) // 远程库配置 new ModuleFederationPlugin({ name: 'libs', filename: 'remote-entry.js', exposes: { './utils': './src/utils' } }) ``` **收益**: - 构建时间降低约40%(通过共享公共依赖) - 热更新速度提升50%+ #### 2. 编译缓存策略 | 缓存类型 | 存储位置 | 命中率 | 适用场景 | |----------------|---------------|--------|------------------| | HardSource | 磁盘 | 85%↑ | 本地开发环境 | | Vite预编译 | node_modules/.vite | 95%↑ | Vite项目 | | Webpack5缓存 | node_modules/.cache | 70%↑ | 生产构建 | --- ### 四、硬件级加速方案 #### 1. 存储设备优化 ```bash # 查看文件系统性能 dd if=/dev/zero of=./testfile bs=1G count=1 oflag=dsync # 典型结果对比: - HDD: ≈120MB/s - SATA SSD: ≈500MB/s - NVMe SSD: ≈3500MB/s ``` **建议**: - 项目目录必须放在SSD - 禁用杀毒软件实时监控`node_modules` #### 2. 内存配置优化 ```javascript // 增加Node.js内存限制 node --max-old-space-size=8192 node_modules/webpack/bin/webpack.js // webpack配置内存缓存 module.exports = { cache: { type: 'filesystem', memoryCacheUnaffected: true, allowCollectingMemory: true } } ``` **内存计算公式**: 推荐内存 = 项目体积 × 1.5 + 1GB --- ### 五、极限优化手段 #### 1. 编译时与运行时解耦 ```javascript // 使用SWC替代Babel(速度提升5-20倍) module.exports = { module: { rules: [{ test: /\.js$/, use: { loader: 'swc-loader', options: { jsc: { parser: { syntax: 'ecmascript' } } } } }] } } ``` #### 2. 按需编译架构 ```typescript // 动态加载示例(减少首屏编译量) const Login = lazy(() => import(/* webpackPrefetch: true */ './Login')); ``` --- ### 六、生产环境专项优化 ```javascript // 关闭非必要功能 module.exports = { devtool: false, // 禁用SourceMap可节省30%时间 performance: { hints: false // 关闭体积警告 }, stats: 'errors-only' // 减少日志输出 } ``` --- ### 关键指标参考 | 优化措施 | 预期提速 | 实施成本 | 风险 | |-------------------|----------|----------|------| | 升级Webpack5 | 20-30% | 低 | 低 | | 启用SWC编译 | 50-70% | 中 | 中 | | 模块联邦 | 40%↑ | 高 | 高 | | 硬件升级至NVMe SSD| 30-50% | 高 | 无 | --- **实施路线建议**: 1. 先执行`webpack-bundle-analyzer`定位瓶颈 2. 实施"通用加速策略"中的1-4项 3. 升级构建工具到最新版 4. 评估引入SWC/Rust工具链 5. 最后考虑硬件升级 > 注:当项目超过50万行代码时,建议采用**增量编译**+**分布式编译集群**方案
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小满zs

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值