C# NativeAOT生成.so 库供Linux下C++调用

C# NativeAOT生成.so 库供Linux下C++调用

#nativeAOT

✅ 最终目标

使用 C# 编写 Add(int a, int b) → 编译为 libadd.so → 在 Linux C++ 中通过 dlopen + dlsym 调用


🧱 第 1 步:创建 NativeAOT 项目

复制代码
xxxxxxxxxx
复制代码
dotnet new classlib -n AddLibrary
复制代码
cd AddLibrary

📄 修改 AddLibrary.csproj

编辑 AddLibrary.csproj 文件,设置为使用 NativeAOT 编译:

复制代码
xxxxxxxxxx
复制代码
<Project Sdk="Microsoft.NET.Sdk">
复制代码
  <PropertyGroup>
复制代码
    <OutputType>Library</OutputType>
复制代码
    <TargetFramework>net8.0</TargetFramework>
复制代码
    <PublishAot>true</PublishAot>
复制代码
    <SelfContained>true</SelfContained>
复制代码
    <UseNativeAot>true</UseNativeAot>
复制代码
  </PropertyGroup>
复制代码
</Project>

🧠 第 2 步:编写导出函数(Add)

创建或替换 Class1.cs 文件内容如下:
x

复制代码
using System;
复制代码
using System.Runtime.InteropServices;
复制代码
复制代码
public static class Exports
复制代码
{
复制代码
    [UnmanagedCallersOnly(EntryPoint = "Add")]
复制代码
    public static int Add(int a, int b)
复制代码
    {
复制代码
        return a + b;
复制代码
    }
复制代码
}

🛠 第 3 步:在 Linux 或 WSL 中发布为 .so(WSL为例)

🛠️ 安装WSL

复制代码
xxxxxxxxxx
复制代码
wsl --install

🛠️WSL下安装配置dotnet库

复制代码
xxxxxxxxxx
复制代码
wget https://dotnet.microsoft.com/download/dotnet/scripts/v1/dotnet-install.sh
复制代码
chmod +x dotnet-install.sh
复制代码
./dotnet-install.sh --version 8.0.204

🛠️配置环境变量

你只需在终端执行(或添加到 ~/.bashrc / ~/.zshrc):

复制代码
xxxxxxxxxx
复制代码
export DOTNET_ROOT=$HOME/.dotnet export PATH=$PATH:$HOME/.dotnet:$HOME/.dotnet/tools

然后运行:

复制代码
xxxxxxxxxx
复制代码
source ~/.bashrc  # 或 ~/.zshrc,如果你用的是 zsh

确认安装成功:

复制代码
xxxxxxxxxx
复制代码
dotnet --version

✅ 输出 8.0.204 即成功!

🛠️安装构建环境

复制代码
xxxxxxxxxx
复制代码
sudo apt update
复制代码
sudo apt install -y build-essential clang zlib1g-dev

解释:

  • build-essential:包含 gcc, g++, make, ld 等工具
  • clang:LLVM 链接器(NativeAOT 优先使用)
  • zlib1g-dev:NativeAOT 部分场景下需要

验证是否安装成功

复制代码
xxxxxxxxxx
复制代码
gcc --version
复制代码
clang --version

必须在 Linux 或 WSL 环境下 执行: 示例:项目位于 d:/Work/Test/NativeAOT/ULib

复制代码
xxxxxxxxxx
复制代码
cd /mnt/d/Work/Test/NativeAOT/ULib
复制代码
dotnet publish -c Release -r linux-x64

输出文件位于:

复制代码
xxxxxxxxxx
复制代码
bin/Release/net8.0/linux-x64/native/AddLibrary.so

可重命名为:

复制代码
xxxxxxxxxx
复制代码
cp bin/Release/net8.0/linux-x64/native/AddLibrary.so ./libadd.so

✅ 第 4 步:用 C++ 调用 libadd.so

创建一个 main.cpp

复制代码
xxxxxxxxxx
复制代码
#include <iostream>
复制代码
#include <dlfcn.h>
复制代码
复制代码
typedef int (*AddFunc)(int, int);
复制代码
复制代码
int main() {
复制代码
    //加载当前目录下的libadd.so
复制代码
    void* handle = dlopen("./libadd.so", RTLD_NOW);
复制代码
    if (!handle) {
复制代码
        std::cerr << "Failed to load .so: " << dlerror() << std::endl;
复制代码
        return 1;
复制代码
    }
复制代码
复制代码
    AddFunc add = (AddFunc)dlsym(handle, "Add");
复制代码
    if (!add) {
复制代码
        std::cerr << "Failed to find symbol: " << dlerror() << std::endl;
复制代码
        dlclose(handle);
复制代码
        return 2;
复制代码
    }
复制代码
复制代码
    int result = add(10, 20);
复制代码
    std::cout << "✅ Add(10, 20) = " << result << std::endl;
复制代码
复制代码
    dlclose(handle);
复制代码
    return 0;
复制代码
}

🧪 第 5 步:编译并运行 C++ 调用测试

确保在main.cpplibadd.so 所在的目录中运行

复制代码
xxxxxxxxxx
复制代码
g++ main.cpp -o testadd -ldl
复制代码
./testadd

输出:

复制代码
xxxxxxxxxx
复制代码
✅ Add(10, 20) = 30

📦 最终结构示意

复制代码
xxxxxxxxxx
复制代码
AddLibrary/
复制代码
├── AddLibrary.csproj
复制代码
├── Class1.cs  (含 Add 方法)
复制代码
├── bin/
复制代码
│   └── Release/...
复制代码
├── libadd.so  <-- 编译后复制出来
复制代码
└── main.cpp   <-- 调用测试

✅ 额外建议

目标 操作
自动编译 .so + 测试 build.sh 脚本
发布 Windows .dll -r win-x64
多函数导出 增加多个 [UnmanagedCallersOnly(EntryPoint = "...")]
高性能 避免复杂对象,使用基本类型指针(int, float*)

✅ 总结命令回顾

复制代码
dotnet new classlib -n AddLibrary
复制代码
# 编辑代码 & csproj
复制代码
dotnet publish -c Release -r linux-x64
复制代码
g++ main.cpp -o testadd -ldl
复制代码
./testadd
相关推荐
cplmlm14 小时前
EF Core使用CodeFirst生成postgresql数据库表名以及字段名用蛇形命名法,而类名仍使用驼峰命名
c#
lingxiao1688816 小时前
WebApi详解+Unity注入--下篇:Unity注入
unity·c#·wpf
lingxiao1688818 小时前
WebApi详解+Unity注入--中篇:.net core的WebAPI
unity·c#·.netcore
ServBay19 小时前
C# 成为 2025 年的编程语言,7个C#技巧助力开发效率
后端·c#·.net
故事不长丨1 天前
C#进制转换:从基础原理到实战应用
开发语言·c#·进制转换·16进制·2进制·10进制
liulilittle1 天前
VEthernet 框架实现 tun2socks 的技术原理
网络·windows·c#·信息与通信·通信
云草桑1 天前
.net AI API应用 客户发的信息提取对接上下游系统报价
ai·c#·.net·semantickernel·sk
故事不长丨1 天前
C#File文件操作全解析:从基础用法到异常处理
服务器·开发语言·visualstudio·c#·文件操作·io流·file
工程师0071 天前
C# 动态编程(基于 dynamic 类型)
开发语言·c#·dynamic·动态编程
用户298698530141 天前
C#: 在Word文档中添加或移除可编辑区域
后端·c#