想象一下你正在构建一个高性能的 Web 应用程序,比如一个 3D 游戏、一个图像编辑器或者一个数据处理仪表盘。你需要应用程序快速流畅,在不减速的情况下执行各种复杂操作。但是仅靠 JavaScript 能做到的有限:无论你如何优化,总会有 JavaScript 无法快速运行的代码。

WebAssembly(Wasm)—— 有了这项酷炫的技术,现在我们可以在浏览器中运行高性能代码,几乎就像在原生应用程序中一样。

WebAssembly 究竟是什么

WebAssembly 或 Wasm 到底是什么呢?它基本上是 JavaScript 的超级伙伴。WebAssembly 是一种低级二进制格式,可以在浏览器中以接近原生的速度运行。它是为计算密集型任务而构建的,而这些任务 JavaScript 自己处理起来效果不太好。

最棒的是,WebAssembly 与特定编程语言无关。它是一个与语言无关的平台,像 C、C++ 或 Rust 等语言的代码可以直接在浏览器中运行。开发者终于可以将其他语言的高性能代码编译成 WebAssembly,在 Web 上与 JavaScript 一起使用。

WebAssembly 特性

极速性能
Wasm 代码的运行速度几乎与原生应用程序一样快,所以你可以将它用于性能密集型任务。如果你正在构建一个图像编辑器,Wasm 可以轻松处理实时图像处理,如调整大小、颜色调整或应用滤镜,而让 JavaScript 处理用户界面。

跨浏览器一致
所有主流浏览器都支持 WebAssembly,即 Chrome、Firefox、Safari 和 Edge。这意味着无论你的用户在哪里,Wasm 代码的运行方式都相似。所以,我们可以保证应用程序的性能始终一致且快速。

更多语言选择
有了 WebAssembly,你不限于使用 JavaScript。你可以引入其他语言,如 C++ 或 Rust,它们以性能和内存效率著称。这对于速度至关重要的项目,或者当你想重用现有代码库时非常有用。

优化资源使用
WebAssembly 被开发为低内存占用。这使得它适合资源有限的设备,如移动设备。这一点非常重要,因为现代应用程序需要在各种设备上运行。

何时应该使用 WebAssembly

并非每个 Web 项目都需要 WebAssembly。对于许多事情,JavaScript 仍然完全够用:表单验证、基本交互、DOM 操作…… 但如果你需要更快的速度,或者你正在处理特别大量的数据,以下是 Wasm 可能派上用场的时候:

  • 图形密集型应用程序:需要 3D 渲染的应用程序,如基于 Web 的游戏或模拟等
  • 实时数据处理:需要快速计算的应用程序,如金融 / 科学分析工具等
  • Web 上的遗留代码:如果你有现有的用 C++ 或 Rust 编写的代码,WebAssembly 允许你将其带到 Web 上而无需完全重写。

举个栗子

我所在的团队就开发了一款类似于 figma 的图形编辑软件,其底层的渲染引擎则是由 c++ 实现,然后通过编译成 WebAssembly 后提供给 js 层调用,从而保障丝滑的操作体验。

开发一个 WebAssembly 应用

让我们通过一个简单的例子来看看 WebAssembly 如何与 JavaScript 交互。我们将用 C 编写一个计算斐波那契数列的函数(当数字较大时,这是一项计算密集型任务),然后从 JavaScript 中调用它。

步骤一、用 C 编写函数

#include <stdint.h>

uint32_t fibonacci(uint32_t n) {
    if (n <= 1) return n;
    return fibonacci(n - 1) + fibonacci(n - 2)
}

步骤二、编译

要将 C 编译为 WebAssembly,你需要 Emscripten,这是一个将 C/C++ 代码转换为 Wasm 的工具。安装 Emscripten 后,使用以下命令:

emcc fibonacci.c -s WASM=1 -o fibonacci.js

此命令将生成两个文件:

  • fibonacci.wasm:WebAssembly 二进制文件
  • fibonacci.js:一个用于加载 WebAssembly 模块的 JavaScript 文件

步骤三、在 JavaScript 中使用 WebAssembly

创建一个 HTML 文件来加载并运行 WebAssembly 代码:

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <title>WebAssemblyDemo</title>
</head>

<body>
  <h1>Fibonacci Sequence with webAssembly</h1>
  <button onclick="calculateFibonacci()">Calculate Fibonacci</button>
  <p id="result"></p>
  <script src="fibonacci.js"></script>
</body>

</html>

当你在浏览器中打开这个文件并点击按钮时,WebAssembly 将高效地计算斐波那契数列,而不会给 JavaScript 带来压力。

WebAssembly 和 JavaScript:完美搭档

WebAssembly 被设计为 JavaScript 的补充,而不是替代品。它们协同工作效果最佳:Wasm 进行繁重的计算工作,JavaScript 处理用户界面逻辑和浏览器交互。

例如场景:数据可视化
想象你正在构建一个数据可视化应用程序,需要处理非常大的数据集。WebAssembly 可以处理数据处理和代码中其他对性能敏感的计算部分,而 JavaScript 仍然可以用于渲染图表等事情,在性能和交互性之间提供良好的平衡。

展望

通过在浏览器中实现接近原生的性能,我们终于可以构建那些以前仅限于原生桌面平台的雄心勃勃的应用程序和体验。

对于前端开发者来说,WebAssembly 提供了一个机会,可以开始处理比仅使用 JavaScript 性能更好的应用程序。