百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

如何在C#中调用C++方法(c#调用c++的类)

liuian 2025-07-07 20:10 3 浏览

主要方式

C#主要通过两种方式提供对非托管代码的调用,第一种是使用平台调用(Platform Invoke,P/Invoke),第二种是使用不安全代码(unsafe),日常开发中我们使用最多的就是第一种,所以下面我们来介绍下如何使用平台调用。

平台调用

CLI通过P/Invoke提供该功能,它允许对非托管DLL所导出的函数执行API调用。平台调用主要有这些步骤:声明外部函数、映射数据类型、错误处理。

1. 声明外部函数

和类的所有普通方法一样,我们需要在类的上下文中声明目标API,同时为它添加extern修饰符,从而把它声明为外部函数。它几乎总是静态的(static)。如下代码所示,我们声明了外部函数SendMessage(位于user32.dll中),EntryPoint属性可以省略不写,也可以指定一个函数名用来覆盖默认的行为,CharSet属性则用于指定调用API的Unicode还是ASCII版本或者由.NET平台自动尝试调用。

using System;
using System.Runtime.InteropServices;

namespace PayPlatformProxy
{
    public class LevyTest
    {
        [DllImport("user32.dll", EntryPoint = "SendMessage", CharSet = CharSet.Auto)]
        private static extern IntPtr SendMessage(IntPtr hWnd, int Msg, IntPtr wParam, IntPtr lParam);
    }
}

2. 数据类型映射

  • 托管代码的基元类型不会随处理器改变大小,无论是32位还是64位处理器,int类型始终是32位,但在非托管代码中,内存指针会随处理器而变化。因此,C++中的HANDLE和LPVOID等类型不能对应为int,而应该对应为IntPtr,其大小随处理器内存布局而变化。
  • 微软官网文档--类型封送有涉及类型的映射,大家可以参考学习下。另外,pinvoke.net中也有很多系统API在C#中的声明示例,输入关键字看看示例是如何映射,从而完成非托管类型到托管类型的映射。
  • 此外,通过工具pinvoke-interop-assistant也可以来生成C++方法在C#中对应的签名,实例演示部分我们将使用此工具。

实例演示

如下我们定义了C++的方法LevyAgeAdd,它的目的是输入姓名和年龄然后返回新的年龄(newAge)和提示年龄增长的信息(resultMsg)。

extern "C"
{
   _declspec(dllexport) void LevyAgeAdd(const char* name, int age, int* newAge, char* resultMsg);
}

下面我们在pinvoke-interop-assistant工具中输入C++方法的声明,然后点击Generate就能生成对应的C#方法的签名了,如下图所示

接着我们通过C#代码来调用此C++代码,首先拷贝出外部函数LevyAgeAdd的声明,然后调用方法传入各个参数值,如下:

[DllImport("Project1.dll", EntryPoint="LevyAgeAdd", CharSet = CharSet.Auto)]
public static extern void LevyAgeAdd([In()][MarshalAs(UnmanagedType.LPStr)] string name, int age, ref int newAge, IntPtr resultMsg);

void Main()
{
 int newAge = 0;
 IntPtr p = Marshal.AllocHGlobal(255);
 //调用C++方法
 LevyAgeAdd("杨小豆", 25, ref newAge, p);
 string resultMsg = Marshal.PtrToStringAnsi(p);
 #34;newAge的值: {newAge}".Dump();
 #34;resultMsg的值: {resultMsg}".Dump();
}

输出结果如下图:

哈哈,调用结果正确!真香!事实上我们C++代码的实现也确实是把年龄增加了3岁然后输出提示信息的。希望本文能对童鞋们有所帮助!

相关推荐

C/C++恶意代码盘点(一):进程遍历丨木马病毒丨密码记录

恶意代码的分类包括计算机病毒、蠕虫、木马、后门、Rootkit、流氓软件、间谍软件、广告软件、僵尸(bot)、Exploit等等,有些技术经常用到,有的也是必然用到。恶意代码常见功能技术如下:进程遍...

跨越十年的C++演进:C++11新特性全解析

原作者:Linux教程,原文「链接」:https://mp.weixin.qq.com/s/oFbiFlqiwgVcJIMMvTelEA很多刚刚进入C++领域的朋友,最初是从C语言转过来的。因...

如何在C#中调用C++方法(c#调用c++的类)

主要方式C#主要通过两种方式提供对非托管代码的调用,第一种是使用平台调用(PlatformInvoke,P/Invoke),第二种是使用不安全代码(unsafe),日常开发中我们使用最多的就是第一种...

C语言字符数组和字符串(c语言字符数组和字符串数组)

用来存放字符的数组称为字符数组,例如:charc[10];字符数组也可以是二维或多维数组。例如:charc[5][10];字符数组也允许在定义时进行初始化,例如:charc[10]={'c',...

C语言指针,如何操作字符串?linux C第45讲

1指针操作字符串在学习数组的时候,我们了解了字符串的定义,我们可以定义一个字符数组,用来存放一个字符串,例如:chararray[]={"abcde"};charbuf[]=...

信奥赛C++常用的算法总结(信息学奥赛c语言和c++有什么区别)

1、桶排序核心:①创建盛下所有数的数组②将每个数作为编号放入桶里优点:稳定、简单、容易考缺点:空间复杂度较大,时间复杂度较大。#include<bits/stdc++.h>usingn...

Arduino 使用 C 字符串(arduino 字符串 数组)

问题您想了解如何使用原始字符字符串:创建字符串、查找其长度以及比较、复制或附加字符串。核心C语言不支持Arduino样式的String功能,因此您想了解针对基本字符数组编写的其他平台的代码...

C++20 四大特性之一:Module 特性详解

C++20最大的特性是什么?最大的特性是迄今为止没有哪一款编译器完全实现了所有特性。文章来源:网易云信有人认为C++20是C++11以来最大的一次改动,甚至比C++11还要大。本文仅介绍...

C/C++的const常量总结(c++语言const)

“所谓常量,就是在代码运行过程中值恒定不变的标识符,该标识符的值可以是一个常数,也可以是字符串。”在C/C++中,通常使用define宏定义或者const来定义常量,比如:#definePI3....

学习分享 | 通过C++python的对比,帮你快速入门python

一、前言对于只接触过静态语言,而从未使用过动态语言的人来说,第一次看到python的语法可能会大为惊叹。不用申明变量类型?不用等老半天编译完成就能直接运行?不用小心的维护指针?还不用写CMakelis...

字符串常量,C语言字符串常量详解

字符常量是由一对单撇号括起来的单个字符,如'a'、'D'、'?'、'#39;。在C语言中,除了字符常量外还有字符串常量,顾名思义就是多个“...

C/C++中的内存四区(c++的内存区域分为)

1代码区存放CPU执行的机器指令。通常代码区是可共享的(即另外的执行程序可以调用它),使其可共享的目的是对于频繁被执行的程序,只需要在内存中有一份代码即可。代码区通常是只读的,使其只读的原因是防...

通过pybind11来实现python调用C++接口(一)

有小伙伴很好奇,怎么样实现python调用C++接口?哈哈,手把手教程来了。第一步:我们需要安装pybind11这个纯头文件的库,目前该库支持c++11及以上版本,在你的环境中通过命令行输入:apt-...

深入了解C++如何注释以及在哪儿注释-开课吧广场

注释虽然写起来很痛苦,但对保证代码可读性至关重要,同时这也是每一个C++开发工程师所需要做好的事情。那么C++开发过程中该如何注释?应该在哪写注释呢?关于注释风格,很多C++的Coders更喜欢行注释...

C++核心知识点速查手册(实用重点版)

一、基础必备核心1.指针与引用(遥控器原理)指针:存储地址的变量(像电视遥控器)inttv=100;//电视机int*remote=&tv;//遥控器指向电视*...