0

0

C# WebAssemblytime (Wasmtime) for .NET方法 C#如何运行WASM模块

煙雲

煙雲

发布时间:2026-01-31 10:03:09

|

376人浏览过

|

来源于php中文网

原创

Wasmtime是用Rust编写的独立WebAssembly运行时,与.NET无官方集成;它不提供C#绑定,需通过P/Invoke调用其C API,且无原生.NET支持。

c# webassemblytime (wasmtime) for .net方法 c#如何运行wasm模块

Wasmtime 是什么,和 .NET 的关系是什么

Wasmtime 是一个用 Rust 编写的独立 WebAssembly 运行时,它本身不是 .NET 库,也不直接支持 C# 调用 WASM 模块。你不能在 .NET(包括 Blazor WebAssembly)里直接 new Wasmtime.Engine() —— 因为官方没有提供 .NET 绑定。目前 Wasmtime 官方只提供 C API 和 Rust、Python、Go 等语言的绑定,没有官方 C#/.NET SDK

这意味着:

  • 你想在 C# 中运行 WASM,不能靠 NuGet 上搜 Wasmtime 就开干
  • Wasmtime 的二进制(如 wasmtime.dlllibwasmtime.so)需要手动 P/Invoke 调用,且得自己维护 ABI 兼容性
  • .NET 6+ 原生支持的是 WebAssemblyHostBuilder(用于 Blazor),但那是运行 .NET 代码编译成 WASM,不是“运行任意 WASM 模块”

当前可行的 C# 运行 WASM 方案有哪些

真正能在生产中考虑的路径只有两条,取决于你的场景:

  • 如果你要从 .NET(如 ASP.NET Core 后端)加载并执行第三方 .wasm 文件

    • P/Invoke 调用 wasmtime-c-api(需自己编译或下载预编译的 libwasmtime
    • 必须手动管理 EngineStoreModuleInstance 生命周期
    • 导出函数调用需按 C ABI 处理参数(例如 i32 映射到 intf64double),字符串需通过线性内存读写
  • 如果你只是想在 Blazor WebAssembly 前端跑 WASM 逻辑

    • 不要用 Wasmtime —— 浏览器原生支持 WebAssembly.instantiate()
    • 更现实的做法是用 JavaScript 互操作(IJSRuntime)加载并调用 WASM,再把结果传回 C#
    • 或者直接用 WASI 兼容的工具链(如 wasmtime CLI)做离线测试,不嵌入 .NET 进程

常见错误现象包括:

  • DllNotFoundException: libwasmtime:没把对应平台的 native 库(x64/arm64)放进 runtimes/ 目录,也没设 CopyToOutputDirectory
  • AccessViolationException:C# 传了托管内存指针给 WASM,但 Wasmtime 要求所有数据都在其 Store.Memory
  • 函数导出找不到:WASM 模块没导出(export "add"),或 C# 用 GetExport 时名字大小写/下划线不一致

有没有更轻量、.NET 友好的替代方案

有,而且更稳:

奇布塔
奇布塔

基于AI生成技术的一站式有声绘本创作平台

下载
  • Wasmtime 太重,而 WasmerWasmEdge 同样没有官方 .NET 绑定
  • 目前唯一有活跃 .NET 支持的是 WatsonWebsocket 风格的轮子?不,其实不是——真正可用的是:CoreWasm(纯 C# WASM 解释器,性能低但完全可控)和 Wasmtime.NET(非官方社区绑定,GitHub 上有,但更新慢、不支持 WASI、仅限简单算术)

如果你只要跑数学计算或配置解析类 WASM:

  • CoreWasm NuGet 包(CoreWasm.Runtime),支持 WASM 二进制加载、简单导入/导出
  • 但别指望它跑 ffmpeg.wasm 或带内存分配的模块——它没实现 __heap_base__data_end 等链接约定

性能与兼容性影响明显:

  • CoreWasm 是解释执行,比 native Wasmtime 慢 10–100×
  • Wasmtime + P/Invoke 能接近原生速度,但跨平台打包麻烦(Windows/Linux/macOS 需不同 native 库)
  • 所有方案都不支持 WASI 文件系统、网络等高级功能,除非你自己实现 fd_read 等导入函数

实际调用 Wasmtime 的最小可行代码长什么样

假设你已把 libwasmtime.so(Linux)放在输出目录,并用 [DllImport("libwasmtime")] 声明了函数:

[DllImport("libwasmtime")]
private static extern IntPtr wasmtime_engine_new();

[DllImport("libwasmtime")] private static extern IntPtr wasmtime_store_new(IntPtr engine, IntPtr context);

[DllImport("libwasmtime")] private static extern IntPtr wasmtime_module_new(IntPtr store, byte* wasmBytes, UIntPtr wasmLen);

然后你要:

  • Marshal.AllocHGlobal 把 WASM 字节数组复制到 unmanaged 内存
  • 调用 wasmtime_module_new 得到 Module*
  • 再调用 wasmtime_instance_new,传入导入表(必须是 wasmtime_importtype_t* 数组)
  • 最后用 wasmtime_instance_get_export 拿函数指针,再用 Marshal.GetDelegateForFunctionPointer 转成 C# 委托

这整套流程没有封装,每一步都容易崩。真正上线前,你得写完整错误码检查(wasmtime_error_t*)、GC 保护(避免 Store 被回收)、线程安全(wasmtime 默认非线程安全)。

复杂点不在语法,而在内存模型对齐和生命周期管理——WASM 的线性内存、.NET 的 GC 堆、native 运行时的 Store,三者边界稍一模糊就会 crash。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
C++系统编程内存管理_C++系统编程怎么与Rust竞争内存安全
C++系统编程内存管理_C++系统编程怎么与Rust竞争内存安全

C++系统编程中的内存管理是指 对程序运行时内存的申请、使用和释放进行精细控制的机制,涵盖了栈、堆、静态区等不同区域,开发者需要通过new/delete、智能指针或内存池等方式管理动态内存,以避免内存泄漏、野指针等问题,确保程序高效稳定运行。它核心在于开发者对低层内存有完全控制权,带来灵活性,但也伴随高责任,是C++性能优化的关键。

10

2025.12.22

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1503

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

625

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

655

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

610

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

172

2025.07.29

php环境变量如何设置
php环境变量如何设置

本合集详细讲解PHP环境变量的设置方法,涵盖Windows、Linux及常见服务器环境配置技巧,助你快速掌握环境变量的正确配置。阅读专题下面的文章了解更多详细内容。

0

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.2万人学习

Git 教程
Git 教程

共21课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号