Transformers</h1><p class="page-description"></p><table class="properties"><tbody><tr class="property-row property-row-created_by"><th><span class="icon property-icon"><svg role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesCreatedBy"><path d="M8 15.126C11.8623 15.126 15.0615 11.9336 15.0615 8.06445C15.0615 4.20215 11.8623 1.00293 7.99316 1.00293C4.13086 1.00293 0.938477 4.20215 0.938477 8.06445C0.938477 11.9336 4.1377 15.126 8 15.126ZM8 10.4229C6.05176 10.4229 4.54785 11.1133 3.83008 11.9131C2.90039 10.9082 2.33301 9.55469 2.33301 8.06445C2.33301 4.91992 4.84863 2.39746 7.99316 2.39746C11.1377 2.39746 13.6738 4.91992 13.6738 8.06445C13.6738 9.55469 13.1064 10.9082 12.1699 11.9131C11.4521 11.1133 9.94824 10.4229 8 10.4229ZM8 9.30176C9.32617 9.30859 10.3516 8.18066 10.3516 6.71094C10.3516 5.33008 9.31934 4.18164 8 4.18164C6.6875 4.18164 5.6416 5.33008 5.64844 6.71094C5.65527 8.18066 6.68066 9.28809 8 9.30176Z"></path></svg></span>Created by</th><td><span class="user"><img src="JavaScript%20Bundlers%20&amp;%20Transformers%2004214428f78e4a53a378c95c83bea343/IMG_2295.jpg" class="icon user-icon"/>JiaLin Huang</span></td></tr><tr class="property-row property-row-last_edited_time"><th><span class="icon property-icon"><svg role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesCreatedAt"><path d="M8 15.126C11.8623 15.126 15.0615 11.9336 15.0615 8.06445C15.0615 4.20215 11.8623 1.00293 7.99316 1.00293C4.13086 1.00293 0.938477 4.20215 0.938477 8.06445C0.938477 11.9336 4.1377 15.126 8 15.126ZM8 13.7383C4.85547 13.7383 2.33301 11.209 2.33301 8.06445C2.33301 4.91992 4.84863 2.39746 7.99316 2.39746C11.1377 2.39746 13.6738 4.91992 13.6738 8.06445C13.6738 11.209 11.1445 13.7383 8 13.7383ZM4.54102 8.91211H7.99316C8.30078 8.91211 8.54004 8.67285 8.54004 8.37207V3.8877C8.54004 3.58691 8.30078 3.34766 7.99316 3.34766C7.69238 3.34766 7.45312 3.58691 7.45312 3.8877V7.83203H4.54102C4.2334 7.83203 4.00098 8.06445 4.00098 8.37207C4.00098 8.67285 4.2334 8.91211 4.54102 8.91211Z"></path></svg></span>Last edited</th><td><time>@2024年10月29日 14:44</time></td></tr><tr class="property-row property-row-multi_select"><th><span class="icon property-icon"><svg role="graphics-symbol" viewBox="0 0 16 16" style="width:14px;height:14px;display:block;fill:rgba(55, 53, 47, 0.45);flex-shrink:0" class="typesMultipleSelect"><path d="M1.91602 4.83789C2.44238 4.83789 2.87305 4.40723 2.87305 3.87402C2.87305 3.34766 2.44238 2.91699 1.91602 2.91699C1.38281 2.91699 0.952148 3.34766 0.952148 3.87402C0.952148 4.40723 1.38281 4.83789 1.91602 4.83789ZM5.1084 4.52344H14.3984C14.7607 4.52344 15.0479 4.23633 15.0479 3.87402C15.0479 3.51172 14.7607 3.22461 14.3984 3.22461H5.1084C4.74609 3.22461 4.45898 3.51172 4.45898 3.87402C4.45898 4.23633 4.74609 4.52344 5.1084 4.52344ZM1.91602 9.03516C2.44238 9.03516 2.87305 8.60449 2.87305 8.07129C2.87305 7.54492 2.44238 7.11426 1.91602 7.11426C1.38281 7.11426 0.952148 7.54492 0.952148 8.07129C0.952148 8.60449 1.38281 9.03516 1.91602 9.03516ZM5.1084 8.7207H14.3984C14.7607 8.7207 15.0479 8.43359 15.0479 8.07129C15.0479 7.70898 14.7607 7.42188 14.3984 7.42188H5.1084C4.74609 7.42188 4.45898 7.70898 4.45898 8.07129C4.45898 8.43359 4.74609 8.7207 5.1084 8.7207ZM1.91602 13.2324C2.44238 13.2324 2.87305 12.8018 2.87305 12.2686C2.87305 11.7422 2.44238 11.3115 1.91602 11.3115C1.38281 11.3115 0.952148 11.7422 0.952148 12.2686C0.952148 12.8018 1.38281 13.2324 1.91602 13.2324ZM5.1084 12.918H14.3984C14.7607 12.918 15.0479 12.6309 15.0479 12.2686C15.0479 11.9062 14.7607 11.6191 14.3984 11.6191H5.1084C4.74609 11.6191 4.45898 11.9062 4.45898 12.2686C4.45898 12.6309 4.74609 12.918 5.1084 12.918Z"></path></svg></span>Tags</th><td></td></tr></tbody></table></header><div class="page-body"><p class="">
</p><p class="">I’ve been using Next.js for a long time, from having to configure everything to now almost zero setup, so I took some time to explore the evolution of JavaScript ecosystem tools.</p><p class="">Nowadays, we almost don&#x27;t need to write a bunch of configuration files.</p><p class="">I also looked at some of the differences between transformers, bundlers, and so on.</p><p class="">Currently, Next.js uses: turbopack for bundling, and swc for transformation.</p><p class="">Vite seems to be used more in Vue projects.</p><p class="">I also reviewed some tool comparisons and benchmarks.</p><p class=""><mark class="highlight-red"><strong>This article doesn’t flow well at all.</strong></mark></p><p class="">…</p><h1 class="">TL;DR</h1><ul class="bulleted-list"><li style="list-style-type:disc">Some tools can act as both transformers and bundlers - at first it&#x27;s separated, but after digging deeper, they tend to be ambiguous.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">SWC is often compared to Babel and is generally classified as a transformer.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Turbopack is often compared to webpack, as well as esbuild, vite, parcel, rollup, etc. - these are generally classified as bundlers.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Starting from Next.js 12, the transformer used is SWC (written in Rust).<p class=""><a href="https://nextjs.org/docs/architecture/nextjs-compiler">https://nextjs.org/docs/architecture/nextjs-compiler</a></p></li></ul><ul class="bulleted-list"><li style="list-style-type:disc">For Next.js 14, the next dev command uses Turbopack (written in Rust).<p class=""><a href="https://nextjs.org/blog/next-14#nextjs-compiler-turbocharged">https://nextjs.org/blog/next-14#nextjs-compiler-turbocharged</a></p></li></ul><p class=""><div class="indented"><p class="">
</p><p class="">
</p></div></p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">SWC (Rust)</h1><ul class="bulleted-list"><li style="list-style-type:disc">Created to replace Babel.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">The author later joined Vercel, and Next.js started using it as the default compiler from version 12 (but it can conflict with Jest and fail to upgrade).</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Deno (another Node.js) also uses SWC to speed up TypeScript startup.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Turbopack and Parcel both use SWC (both written in Rust).</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Webpack uses Babel, but can also use the <code>swc-loader</code>.</li></ul><p class="">
</p><p class=""><a href="https://swc.rs/#features">https://swc.rs/#features</a></p><h3 class="">Features</h3><ul class="bulleted-list"><li style="list-style-type:disc">Compilation</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Bundling (<code>swcpack</code>)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Minification </li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Transforming with WebAssembly</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Usage inside webpack (<code>swc-loader</code>)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Improving Jest performance (<code>@swc/jest</code>)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Custom Plugins</li></ul><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Typescript</h1><p class="">Many tools have built-in TypeScript transpilation capabilities. 3-5 are bundling tools, and they may use Babel or SWC as the transpilation tool, usually as an expanded plugin. </p><p class="">The TypeScript to JavaScript conversion is done first, then Babel or SWC transforms it into browser-understandable JavaScript.</p><p class="">Before, I didn&#x27;t understand that TypeScript is not an independent tool for TS-&gt;JS conversion, but rather for better IDE DX. The actual TS-&gt;JS conversion is handled by the various bundling tools.</p><ol type="1" class="numbered-list" start="1"><li>Babel with @babel/preset-typescript<ul class="toggle"><li><details open=""><summary>.babelrc</summary><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-JSON">{
  &quot;presets&quot;: [
    &quot;@babel/preset-env&quot;,
    &quot;@babel/preset-typescript&quot;
  ]
}</code></pre></details></li></ul></li></ol><ol type="1" class="numbered-list" start="2"><li>esbuild<blockquote class="">JavaScript, CSS, TypeScript, and JSX built-in</blockquote><p class="">zero config ☺️</p></li></ol><ol type="1" class="numbered-list" start="3"><li>Webpack with <code>ts-loader</code> and <code>babel-loader</code><ul class="toggle"><li><details open=""><summary>babel-loader only (simple, fast, no type check, only transform)</summary><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-JavaScript">{
  test: /\.tsx?$/,
  use: &#x27;babel-loader&#x27;,
  options: {
    presets: [&#x27;@babel/preset-typescript&#x27;]
  }
}</code></pre></details></li></ul><ul class="toggle"><li><details open=""><summary>use both (slow but reliable, for production)</summary><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-JavaScript">{
  test: /\.tsx?$/,
  use: [
    &#x27;babel-loader&#x27;,
    &#x27;ts-loader&#x27;
  ]
}</code></pre></details></li></ul><ul class="toggle"><li><details open=""><summary>use both with transpileOnly (balancing..)</summary><p class="">&quot;Transpile&quot; means the process involves less work than &quot;Transform&quot;, as <mark class="highlight-red">it skips type checking</mark> and directly converts TS to JS.</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-JavaScript">{
  test: /\.tsx?$/,
  use: [
    &#x27;babel-loader&#x27;,
    {
      loader: &#x27;ts-loader&#x27;,
      options: {
        transpileOnly: true
      }
    }
  ]
}</code></pre></details></li></ul></li></ol><ol type="1" class="numbered-list" start="4"><li>Rollup with <code>@rollup/plugin-typescript</code><ul class="toggle"><li><details open=""><summary>rollup.config.js</summary><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code"><code class="language-JavaScript">// rollup.config.js
import typescript from &#x27;@rollup/plugin-typescript&#x27;;

export default {
  input: &#x27;src/index.ts&#x27;,
  output: {
    // ...
  },
  plugins: [
    typescript({
      tsconfig: &#x27;./tsconfig.json&#x27;
    })
  ]
};</code></pre></details></li></ul></li></ol><ol type="1" class="numbered-list" start="5"><li>Parcel (default uses SWC)<blockquote class="">TypeScript is a typed superset of JavaScript that compiles to JavaScript. Parcel supports TypeScript out of the box without any additional configuration.</blockquote><p class="">zero config, too ☺️</p></li></ol><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Vite</h1><p class="">some pov from official doc.</p><h3 class="">The Problem: Slow Server Start</h3><blockquote class="">Over time we have seen tools like webpack, Rollup and Parcel, which greatly improved the development experience for frontend developers.</blockquote><blockquote class="">When cold-starting the dev server, a bundler-based build setup has to eagerly crawl and build your entire application before it can be served.</blockquote><figure class="image" style="text-align:left"><a href="https://v2.vitejs.dev/assets/bundler.37740380.png"><img style="width:528px" src="https://v2.vitejs.dev/assets/bundler.37740380.png"/></a></figure><blockquote class=""><strong>Vite improves the dev server start time by first dividing the modules in an application into two categories: dependencies and source code.<br/><br/></strong>Vite pre-bundles dependencies using <strong>esbuild</strong>. esbuild is written in Go and pre-bundles dependencies 10-100x faster than JavaScript-based bundlers.</blockquote><figure class="image" style="text-align:left"><a href="https://v2.vitejs.dev/assets/esm.3070012d.png"><img style="width:528px" src="https://v2.vitejs.dev/assets/esm.3070012d.png"/></a></figure><p class="">use esbuild in dev mode</p><h3 class="">Why Not Bundle with esbuild?</h3><blockquote class="">Rollup is more mature and flexible in these regards. That said, we won&#x27;t rule out the possibility of using <code>esbuild</code> for production build when it stabilizes these features in the future.</blockquote><p class="">use Rollup in production mode</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Turbopack (Rust)</h1><p class="">Turbopack is a bundler that uses SWC for the transformation and compilation process, similar to how Webpack uses Babel.</p><ul class="bulleted-list"><li style="list-style-type:disc">Turbopack was written by the Webpack author and the Next.js team, in Rust.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Turbopack currently supports Next.js, and there are plans to integrate it with Svelte in the future.</li></ul><p class=""><a href="https://turbo.build/pack/docs/why-turbopack">https://turbo.build/pack/docs/why-turbopack</a></p><h3 class="">Why Turbopack?</h3><blockquote class="">We migrated away from several JS-based tools. Babel, gone. Terser, gone. Our next target was another JS-based tool, webpack.</blockquote><blockquote class="">A new generation of native-speed bundlers were emerging, but after assessing the bundlers on the market, we decided to build our own. Why?</blockquote><blockquote class="">esbuild doesn’t have a concept of ‘lazy’ bundling - it’s all-or-nothing, unless you specifically target only certain entry points.</blockquote><h3 class="">Bundling vs Native ESM</h3><blockquote class="">Frameworks like <mark class="highlight-red"><strong>Vite use a technique where they don’t bundle application source code in development mode.</strong></mark><mark class="highlight-red"> </mark><strong><mark class="highlight-red">Instead, they rely on the browser’s native ES Modules system.</mark></strong> This approach results in incredibly responsive updates since they only have to transform a single file.</blockquote><blockquote class="">we <strong>wanted Turbopack to bundle the code in the development server.</strong> Turbopack can do it much faster, especially for larger applications, because it is written in Rust and skips optimization work that is only necessary for production.</blockquote><p class="">
</p><p class="">These days, browsers can natively handle ES modules, which lets Vite just update individual files (using esbuild) and notify the browser (over WebSocket) for hot reloading. But as the project grows bigger, all those individual requests can start to become a bottleneck with the native ESM approach.</p><p class="">A traditional bundled setup is still better in a lot of cases - having one bigger file request is usually better than tons of smaller ones, even during development.</p><p class="">That&#x27;s why Turbopack wants to do the bundling step right in the dev server, just like Webpack. </p><p class="">But Turbopack can do it much faster, and it avoids the startup slowness you can get from Vite&#x27;s super granular modularization (especially with vue SFC).</p><h2 class="">Incremental Computation</h2><blockquote class="">There are two ways to make a process faster: do less work or do work in parallel.</blockquote><h2 class="">Lazy bundling</h2><blockquote class="">Early versions of Next.js tried to bundle the <em>entire</em> web app in development mode. We quickly realized that this ‘eager’ approach was <br/>less than optimal. Modern versions of Next.js bundle only the pages requested by the dev server.<br/></blockquote><blockquote class="">esbuild doesn’t have a concept of ‘lazy’ bundling - it’s all-or-nothing, unless you specifically target only certain entry points.</blockquote><p class="">esbuild is crazy fast and great as a bundler, but the downside is it doesn&#x27;t have any caching capabilities. That&#x27;s where Turbopack is trying to step in and fill that gap.</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">Side Note:</p><p class=""><a href="https://github.com/vercel/turbo">https://github.com/vercel/turbo</a></p><p class="">Turbo includes two project</p><ul class="bulleted-list"><li style="list-style-type:disc"><strong>Turbopack:</strong> an incremental bundler (the successor to Webpack)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><strong>Turborepo:</strong> an incremental build system</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">The Turbo engine: a low-level incremental computation and memoization engin</li></ul><blockquote class="">In the future, Turborepo and Turbopack will merge into a single toolchain--Turbo--that can be used as either a bundler or a build system or both.<br/><br/><a href="https://turbo.build/pack/docs/roadmap">https://turbo.build/pack/docs/roadmap</a></blockquote><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">esbuild, Vite, Snowpack, WMR Differences</h1><p class=""><a href="https://css-tricks.com/comparing-the-new-generation-of-build-tools/#h-why-are-these-tools-all-arriving-now">https://css-tricks.com/comparing-the-new-generation-of-build-tools/#h-why-are-these-tools-all-arriving-now</a></p><p class="">These are some of the key differences between the new generation of build tools:</p><ul class="bulleted-list"><li style="list-style-type:disc">esbuild is excellent at bundle size, but lacks support for modern frontend features like HMR and SSR.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">WMR is no longer maintained as of March 2024.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Snowpack is no longer maintained as of April 2022.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><mark class="highlight-red"><strong>Vite is the current winner in this space.</strong></mark></li></ul><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">esbuild, SWC, TSC, Babel Benchmark</h1><p class=""><a href="https://datastation.multiprocess.io/blog/2021-11-13-benchmarking-esbuild-swc-typescript-babel.html">https://datastation.multiprocess.io/blog/2021-11-13-benchmarking-esbuild-swc-typescript-babel.html</a></p><blockquote class="">Bundlers typically use a transformer under the hood and handle turning multiple JavaScript source files into one JavaScript file (a bundle). Some newer transformers like esbuild and swc work as bundlers too. </blockquote><ul class="bulleted-list"><li style="list-style-type:disc">The Babel author later went on to create <strong>Rome</strong>, another transformer, and then worked on <strong>Biome</strong>.<blockquote class="">Biome is the <strong>official</strong> fork of Rome and it will continue to be Rome’s legacy.</blockquote></li></ul><ul class="bulleted-list"><li style="list-style-type:disc">The benchmark shows that esbuild and SWC significantly outperform Babel and TypeScript in terms of performance.</li></ul><p class="">
</p><h1 class="">Minification Benchmark</h1><p class=""><a href="https://github.com/privatenumber/minification-benchmarks#%EF%B8%8F%EF%B8%8F-js-minification-benchmarks">https://github.com/privatenumber/minification-benchmarks#%EF%B8%8F%EF%B8%8F-js-minification-benchmarks</a></p><p class="">The benchmark results indicate:</p><ul class="bulleted-list"><li style="list-style-type:disc">@swc/core and uglify-js take turns being the top performers in different test cases.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">@swc/core and terser also take turns being the top performers in different test cases.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Terser sometimes times out.</li></ul><hr/><ul class="bulleted-list"><li style="list-style-type:disc"><code>terser</code><ul class="bulleted-list"><li style="list-style-type:circle">used by Webpack and Rollup for bundling.</li></ul></li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><code>babel-minify</code><ul class="bulleted-list"><li style="list-style-type:circle">a Babel plugin</li></ul></li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><code>uglify-js</code><ul class="bulleted-list"><li style="list-style-type:circle">Uglify-js is on par with SWC in some cases.</li></ul></li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><code>esbuild</code><ul class="bulleted-list"><li style="list-style-type:circle">used by Vite, Snowpack, and Parcel for bundling.</li></ul></li></ul><ul class="bulleted-list"><li style="list-style-type:disc"><code>swc</code><ul class="bulleted-list"><li style="list-style-type:circle">a strong performer overall.</li></ul></li></ul><ul class="bulleted-list"><li style="list-style-type:disc">There is a separate article that provides the results of comparing different tools for code minification.<p class=""><a href="https://juejin.cn/post/7052238609709006878">https://juejin.cn/post/7052238609709006878</a></p></li></ul><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">Babel</h1><p class="">There&#x27;s a very detailed article explaining Babel, which is the most comprehensive I&#x27;ve seen.</p><p class=""><a href="https://awdr74100.github.io/2020-03-16-webpack-babelloader/">https://awdr74100.github.io/2020-03-16-webpack-babelloader/</a></p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><p class="">
</p><h1 class="">References</h1><p class=""><a href="https://medium.com/dcardlab/%E6%8E%A1%E7%94%A8-swc-%E5%8F%96%E4%BB%A3-babel-%E5%A4%A7%E5%B9%85%E6%8F%90%E5%8D%87%E7%B7%A8%E8%AD%AF%E9%80%9F%E5%BA%A6-802d9cd0db35">https://medium.com/dcardlab/%E6%8E%A1%E7%94%A8-swc-%E5%8F%96%E4%BB%A3-babel-%E5%A4%A7%E5%B9%85%E6%8F%90%E5%8D%87%E7%B7%A8%E8%AD%AF%E9%80%9F%E5%BA%A6-802d9cd0db35</a></p><p class=""><a href="https://eisenbergeffect.medium.com/an-esbuild-setup-for-typescript-3b24852479fe">https://eisenbergeffect.medium.com/an-esbuild-setup-for-typescript-3b24852479fe</a></p><p class=""><a href="https://parceljs.org/languages/typescript/">https://parceljs.org/languages/typescript/</a></p><p class=""><a href="https://webpack.js.org/configuration/other-options/#parallelism">https://webpack.js.org/configuration/other-options/#parallelism</a></p><p class=""><a href="https://turbo.build/pack/docs/core-concepts#function-level-caching">https://turbo.build/pack/docs/core-concepts#function-level-caching</a></p><p class=""><a href="https://v2.vitejs.dev/guide/why.html">https://v2.vitejs.dev/guide/why.html</a></p><p class=""><a href="https://swc.rs/">https://swc.rs/</a></p><p class=""><a href="https://dev.to/132/i-never-need-webpack-or-babel-anymore-3o45">https://dev.to/132/i-never-need-webpack-or-babel-anymore-3o45</a></p><p class=""><a href="https://leerob.com/n/rust">https://leerob.com/n/rust</a></p><p class=""><a href="https://github.com/preactjs/wmr">https://github.com/preactjs/wmr</a></p><p class=""><a href="https://github.com/FredKSchott/snowpack">https://github.com/FredKSchott/snowpack</a></p><p class=""><a href="https://biomejs.dev/blog/annoucing-biome/">https://biomejs.dev/blog/annoucing-biome/</a></p></div></article><span class="sans" style="font-size:14px;padding-top:2em"></span></body>