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
相关推荐
唐青枫3 小时前
C#.NET log4net 详解
c#·.net
Nemo_XP8 小时前
HttpHelper类处理两种HTTP POST请求
c#
lijingguang15 小时前
在C#中根据URL下载文件并保存到本地,可以使用以下方法(推荐使用现代异步方式)
开发语言·c#
¥-oriented16 小时前
【C#中路径相关的概念】
开发语言·c#
ArabySide16 小时前
【WCF】通过AOP实现基于JWT的授权与鉴权的实践
c#·jwt·aop·wcf
xiaowu08016 小时前
C# Task异步的常用方法
c#
阿蒙Amon16 小时前
C# Linq to Objects 详解:集合处理的终极方案
c#·solr·linq
钢铁男儿16 小时前
C# 委托(调用带引用参数的委托)
java·mysql·c#
番茄小能手17 小时前
【全网唯一】C# 纯本地离线文字识别Windows版dll插件
开发语言·c#
葬歌倾城18 小时前
waferMap图像渲染
c#·wpf