服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C#与C++与互操作实例讲解

C#与C++与互操作实例讲解

2022-08-04 09:39zhaotianff C#

在本篇文章里小编给大家整理了关于C#与C++与互操作实例以及相关内容,需要的朋友们可以学习下。

一、C#调用C++库

1、创建C++库

打开VisualStudio,创建一个C++工程,输入项目名称HelloWorldLib

C#与C++与互操作实例讲解

确定,然后下一步。选择应用程序类型为DLL

C#与C++与互操作实例讲解

单击完成,我们就创建好了一个C++库的项目。

这里为了方便,我们直接在HelloWorldLib.cpp里定义函数

C++库导出有两种方式

一、以C语言接口的方式导出

这种方法就是在函数前面加上 extern "C" __declspec(dllexport)

加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。

?
1
2
3
4
5
6
7
8
9
10
#include "stdafx.h"
#include<iostream>
 
extern "C" __declspec(dllexport) void HelloWorld(char* name);
 
 
extern "C" __declspec(dllexport) void HelloWorld(char* name)
{
 std::cout << "Hello World " << name << std::endl;
}

 

二、以模块定义文件的方式导出

在工程上右键,选择添加-》新建项

C#与C++与互操作实例讲解

然后选择代码-》模块定义文件

C#与C++与互操作实例讲解

在Source.def中输入

?
1
2
3
4
LIBRARY
 
EXPORTS
HelloWorld

 

EXPORTS下面就是要导出的函数,这里不需要添加分号隔开,直接换行就行。

此时,我们函数的定义如下

?
1
2
3
4
5
6
7
8
9
10
#include "stdafx.h"
#include<iostream>
 
void HelloWorld(char* name);
 
 
void HelloWorld(char* name)
{
 std::cout <<"Hello World "<< name << std::endl;
}

 

编译,生成dll。这里需要注意的是,如果生成是64位的库,C#程序也要是64位的,否则会报错。

2、使用C#调用

接下来我们新建一个C#控制台项目

C#与C++与互操作实例讲解

打开前面C++库生成的目录,将HelloWorldLib.dll复制到C#工程的Debug目录下。也可以不复制,只需在引用dll的时候写上完整路径就行了。这里我是直接复制到Debug目录下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
using System.Runtime.InteropServices;
 
namespace ConsoleApplication2
{
 class Program
 {
  [DllImport("HelloWorldLib.dll")]
  public static extern void HelloWorld(string name);
 
  //可以通过EntryPoint特性指定函数入口,然后为函数定义别名
 
  [DllImport("HelloWorldLib.dll", EntryPoint = "HelloWorld")]
  public static extern void CustomName(string name);
  static void Main(string[] args)
  {
   HelloWorld("LiLi");
   //跟上面是一样的
   CustomName("QiQi");
  }
 }
}

 

运行程序,结果如下:

C#与C++与互操作实例讲解

这样就成功创建了一个C#可以调用的C++库

下面我们动态调用C++库,这里委托的作用就比较明显了。把委托比喻为C++的函数指针,一点也不为过。

我们在C++库中再新增一个函数GetYear(),用来获取当前年份。

?
1
2
3
4
5
6
7
8
9
int GetYear();
 
int GetYear()
{
 SYSTEMTIME tm;
 GetLocalTime(&tm);
 
 return tm.wYear;
}

 

记得在导出文件中(Source.def)增加GetYear。编译,生成新的DLL

再新建一个C#控制台程序

代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
using System;
using System.Runtime.InteropServices;
 
namespace ConsoleApplication3
{
 
 class Program
 {
  [DllImport("kernel32.dll")]
  public static extern IntPtr LoadLibrary(string lpFileName);
 
  [DllImport("kernel32.dll")]
  public static extern IntPtr GetProcAddress(IntPtr hModule, string lpProcName);
 
  [DllImport("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]
  public static extern bool FreeLibrary(IntPtr hModule);
 
  //声明委托,这里的签名,需要跟C++库中的对应
  delegate int GetYearDelegate();
 
  static void Main(string[] args)
  {
   GetYearDelegate m_fGetYear;
   IntPtr hModule = LoadLibrary("HelloWorldLib.dll");
   if(hModule != IntPtr.Zero)
   {
    IntPtr hProc = GetProcAddress(hModule, "GetYear");
    if(hProc != IntPtr.Zero)
    {
     m_fGetYear = (GetYearDelegate)Marshal.GetDelegateForFunctionPointer(hProc, typeof(GetYearDelegate));
 
     //在这里可以调用
     int year = m_fGetYear();
     Console.WriteLine("年份是:" + year);
    }
   }
  }
 }
}

 

运行结果:

C#与C++与互操作实例讲解

好的,前面函数里面涉及的都是简单数据类型,下面来介绍一下复杂数据类型。这里指的是结构体

在C++库中定义一个GetDate()的函数,代码如下。这里也要记得在导出文件中添加(Source.def)

struct MyDate
{
 int year;
 int month;
 int day;
};

MyDate GetDate();

MyDate GetDate()
{
 SYSTEMTIME tm;
 GetLocalTime(&tm);
 
 MyDate md;
 md.day = tm.wDay;
 md.month = tm.wMonth;
 md.year = tm.wYear;
 return md;
}

 新建一个C#控制台程序,完整代码如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
using System;
using System.Runtime.InteropServices;
 
namespace ConsoleApplication3
{
 struct MyDate
 {
  public int Year;
  public int Month;
  public int Day;
 }
 
 
 class Program
 {
  [DllImport("kernel32.dll")]
  public static extern IntPtr LoadLibrary(string lpFileName);
 
  [DllImport("kernel32.dll")]
  public static extern IntPtr GetProcAddress(IntPtr hModule, string lpProcName);
 
  [DllImport("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]
  public static extern bool FreeLibrary(IntPtr hModule);
 
  delegate IntPtr GetDateDelegate();
 
  static void Main(string[] args)
  {
   GetDateDelegate m_fGetDate;
   IntPtr hModule = LoadLibrary("HelloWorldLib.dll");
 
   if (hModule != IntPtr.Zero)
   {
    IntPtr hProc = GetProcAddress(hModule, "GetDate");
    if (hProc != IntPtr.Zero)
    {
     m_fGetDate = (GetDateDelegate)Marshal.GetDelegateForFunctionPointer(hProc, typeof(GetDateDelegate));
     IntPtr ptr = m_fGetDate();
     if(ptr != IntPtr.Zero)
     {
      MyDate md = (MyDate)Marshal.PtrToStructure(ptr, typeof(MyDate));
      Console.WriteLine("{0}年-{1}月-{2}日",md.Year,md.Month,md.Day);
     }
    }
   }
  }
 }
}

 

运行结果如下:

C#与C++与互操作实例讲解

C#与C++互操作,很重要的一个地方就是,要注意数据类型的对应。有时还需要加上一些限制,

关于C#与C++数据类型对应

可以参考以下链接:

http://www.tuohang.net/article/237687.html

大部分硬件厂商提供的SDK都是需要C++来调用的,有了上面的知识,使用C#来调用一些硬件的SDK就比较容易了。只需要使用C++再进行一次封装就行了。

二、C++调用C#库

这里用到是C++/CLI,就是如何用C++在·NET中编程。就是因为有这个东西的存在,C++才能调用C#的库

下面新建一个C#类库CSharpLib

C#与C++与互操作实例讲解

 以上就是全部知识点内容,感谢大家对服务器之家的支持。

原文链接:https://www.cnblogs.com/zhaotianff/p/8991847.html

延伸 · 阅读

精彩推荐
  • C#c# 适配器模式

    c# 适配器模式

    适配器模式:将一个类的接口转换成客户希望的另一个接口,Adapter使原本由于接口不兼容而不能一起工作的那些类可以一起工作...

    C#教程网7582021-04-14
  • C#c#实现从字符串数组中把数字的元素找出来

    c#实现从字符串数组中把数字的元素找出来

    下面小编就为大家分享一篇c#实现从字符串数组中把数字的元素找出来的方法,希望对大家有所帮助。一起跟随小编过来看看吧...

    杨明波(Leo Yang)6122022-02-15
  • C#浅谈C#下winform和JS的互相调用和传参(webbrowser)

    浅谈C#下winform和JS的互相调用和传参(webbrowser)

    下面小编就为大家带来一篇浅谈C#下winform和JS的互相调用和传参(webbrowser)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来...

    C#教程网5522021-12-13
  • C#C#使用SqlConnection连接到SQL Server的代码示例

    C#使用SqlConnection连接到SQL Server的代码示例

    这篇文章主要介绍了C#使用SqlConnection连接到SQL Server的代码示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要...

    xuanyin2357782022-07-11
  • C#HttpWebRequest实现下载图片至本地

    HttpWebRequest实现下载图片至本地

    这篇文章主要为大家详细介绍了HttpWebRequest实现下载图片至本地,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    C#教程网8982022-02-27
  • C#C#子线程执行完后通知主线程的方法

    C#子线程执行完后通知主线程的方法

    下面小编就为大家带来一篇C#子线程执行完后通知主线程的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    C#教程网7742021-12-13
  • C#C# FileStream读写的文本操作代码分析

    C# FileStream读写的文本操作代码分析

    这篇文章主要分享了个人使用C# FileStream实现的读写的文本操作的小程序,主要是复习下对filestream的理解,希望对大家学习C#能够有所帮助...

    C#教程网6062021-12-02
  • C#C#获取本机IP地址(ipv4)

    C#获取本机IP地址(ipv4)

    本文主要介绍了C#获取本机所有IP地址的方法、C#获取本机IP地址(ipv4)的方法等。具有很好的参考价值。下面跟着小编一起来看下吧...

    LJD泊水10262021-12-29