网站首页 > 基础教程 正文
C#使用过程中经常会遇到和C++联合开发的过程,通过C++编写动态库,封装成dll后再C#中调用,在此做个记录,并供后期查看
一、新建C#控制台项目
打开VisualStudio,新建一个C#控制台项目,项目名称HelloWorldTest
点击下一步,一个空的默认c#项目创建完成
二、创建C++库
在解决方案上右键--添加--新建项目,建一个C++动态链接库工程,输入项目名称TestDll
创建完成后如下:
这里为了方便,我们直接在HelloWorldLib.cpp里定义函数
C++库导出有两种方式
1、以C语言接口的方式导出
这种方法就是在函数前面加上 extern "C" __declspec(dllexport)
加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。
#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;
}
2、以模块定义文件的方式导出
在工程上右键,选择添加-》新建项
然后选择代码-》模块定义文件
在Source.def中输入
LIBRARY
EXPORTS
HelloWorld
EXPORTS下面就是要导出的函数,这里不需要添加分号隔开,直接换行就行。
此时,我们函数的定义如下
#include "stdafx.h"
#include<iostream>
void HelloWorld(char* name);
void HelloWorld(char* name)
{
std::cout <<"Hello World "<< name << std::endl;
}
编译,生成dll。这里需要注意的是,如果生成是64位的库,C#程序也要是64位的,否则会报错
在C#项目中添加引用:同时把C#代码修改为:
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++库中再新增一个函数GetYear(),用来获取当前年份。
int GetYear();
int GetYear()
{
SYSTEMTIME tm;
GetLocalTime(&tm);
return tm.wYear;
}
记得在导出文件中(Source.def)增加GetYear。编译,生成新的DLL
再新建一个C#控制台程序
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++库中定义一个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#控制台程序,完整代码如下
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);
}
}
}
}
}
}
运行结果如下:
猜你喜欢
- 2024-12-30 C++ 在编程语言谱系中的位置及其特点
- 2024-12-30 C++基础知识总结(超详细总结) c++ 基础
- 2024-12-30 [面试]C++中struct和class的区别是什么
- 2024-12-30 C++类和C语言的结构体有什么区别?linux C++第18讲
- 2024-12-30 C与C++利用pragma pack对结构体做成员打包对齐设置
- 2024-12-30 结构体所占内存大小 结构体所占内存大小是所有成员所占的内存字节数计算
- 2024-12-30 结构体的各种使用方法详细讲解-c\c++
- 2024-12-30 C++,结构体和类,结构体和类的区别,struct与class的区别
- 2024-12-30 C++编程:复合数据类型—结构体 c语言结构体复数运算
- 2024-12-30 C#与C++交互开发系列之复杂类型传递与解析
- 最近发表
- 标签列表
-
- gitpush (61)
- pythonif (68)
- location.href (57)
- tail-f (57)
- pythonifelse (59)
- deletesql (62)
- c++模板 (62)
- css3动画 (57)
- c#event (59)
- linuxgzip (68)
- 字符串连接 (73)
- nginx配置文件详解 (61)
- html标签 (69)
- c++初始化列表 (64)
- exec命令 (59)
- canvasfilltext (58)
- mysqlinnodbmyisam区别 (63)
- arraylistadd (66)
- node教程 (59)
- console.table (62)
- c++time_t (58)
- phpcookie (58)
- mysqldatesub函数 (63)
- window10java环境变量设置 (66)
- c++虚函数和纯虚函数的区别 (66)