WebAssembly 是一种新的编码方式,可以在现代的网络浏览器中运行 — 它是一种低级的类汇编语言,具有紧凑的二进制格式,可以接近原生的性能运行,并为诸如 C/C++ 等语言提供一个编译目标,以便它们可以在Web上运行。
WebAssembly 被设计为可以和 JavaScript 一起协同工作 — 通过使用 WebAssembly 的 JavaScript API,你可以把 WebAssembly 模块加载到一个 JavaScript 应用中并且在两者之间共享功能。这允许你在同一个应用中利用 WebAssembly 的性能和威力以及 JavaScript 的表达力和灵活性。
WebAssembly 如何在浏览器中运行,需要了解几个关键概念,这些概念都是一一映射到了WebAssembly的JavaScript API中。
postMessage()
函数)进行共享。一个模块能够像一个 ES2015 的模块一样声明导入和导出。Memory与Table区别: WebAssembly Table 是一个可变大小的带类型(唯一合法的是函数类型)的引用数组,其中的引用可以被 JavaScript 和 WebAssembly 代码存取。然而,Memory 提供的是一个可变大小的带类型的原始字节数组。所以,把引用存储在 Memory 中是不安全。 在C/C++的原生实现中,函数指针是通过函数代码在进程的虚地址空间的原始地址表示的,并且由于前面提到的安全原因,它是不能被直接存储在线性内存(Memory)中的。取而代之的是,函数引用被存储在 Table 之中。它们的整数索引可以存储在线性内存(Memory)中并进行传递。
C/C++ 和 Javascript 区别:
在线 WASM 汇编程序: WasmFiddle、WasmFiddle++、WasmExplorer
Emscripten工具能够将一段C/C++代码,编译出:
Emscripten 是一个 LLVM(底层虚拟机) 生成 JavaScript 的编译器. 它采用 LLVM的字节码 (例如,使用 Clang 从 C/C++ 或者从其他语言生成的字节码) 并将其编译成可在 Web 上面运行的 JavaScript
Emscripten生成的代码(其默认输出格式为 asm.js ,这是 JavaScript 的高度优化子集「变量一律都是静态类型,并且取消垃圾回收机制」)在许多情况下可以以接近原生的速度执行。
Asm.js 是一个规范,它定义了高度可优化的 JavaScript 严格子集。仅允许诸如 while、if、数字、顶级命名函数和其他简单构造之类的东西。它不允许对象、字符串、闭包以及基本上所有需要堆分配的内容。Asm.js 代码在许多方面都类似于C,但是它仍然是完全有效的 JavaScript,可以在所有当前引擎中运行。
一旦 JavaScript 引擎发现运行的是 asm.js,就知道这是经过优化的代码,可以跳过语法分析这一步,直接转成汇编语言。另外,浏览器还会调用 WebGL 通过 GPU 执行 asm.js。这些就是 asm.js 运行较快的原因。
当前还没有内置的方式让浏览器为你获取模块。当前唯一的方式就是创建一个包含你的 WebAssembly 模块二进制代码的 ArrayBuffer
并且使用 WebAssembly.instantiate()
编译它。
#include <stdio.h>
float add (float a, float b) {
return a + b;
}
12345
通过 WasmExplorer 工具(也可以使用上述的 Emscripten)编译后:
(module
(table 0 anyfunc)
(memory $0 1)
(export "memory" (memory $0))
(export "add" (func $add))
(func $add (; 0 ;) (param $0 f32) (param $1 f32) (result f32)
(f32.add
(get_local $0)
(get_local $1)
)
)
)
123456789101112
示例:
<label for="num1"><input id="num1" type="text"></label>
<label for="num2"><input id="num2" type="text"></label>
<input id="compute" type="button" @click="add" value="计算"></input>
<label id="result"></label>
<script>
const $ = (arg) => document.querySelector(arg)
// 创建内存示例(initial:初始大小;maximum:最大大小)
const memory = new WebAssembly.Memory({ initial: 256, maximum: 256 });
const importObj = {
env: {
abortStackOverflow: () => { throw new Error('overflow'); },
table: new WebAssembly.Table({ initial: 0, maximum: 0, element: 'anyfunc' }),
tableBase: 0,
memory: memory,
memoryBase: 1024,
STACKTOP: 0,
STACK_MAX: memory.buffer.byteLength,
}
};
fetch('./add.wasm').then(response =>
// 1. 返回一个可以解析为带类型数组的promise
response.arrayBuffer()
).then(bytes =>
// 2. 编译和实例化带类型数组
WebAssembly.instantiate(bytes, importObj)
).then(wa => {
$('#compute').addEventListener('click', () => {
// 3. 使用通过 WebAssembly.Instance.exports 属性导出的特性
let result = wa.instance.exports.add(+$('#num1').value, +$('#num2').value)
$('#result').innerHTML = result
})
});
</script>
123456789101112131415161718192021222324252627282930313233343536
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有