C#与C++与互操作
一、C#调用C++库
1、创建C++库
打开VisualStudio,创建一个C++工程,输入项目名称HelloWorldLib
确定,然后下一步。选择应用程序类型为DLL
单击完成,我们就创建好了一个C++库的项目。
这里为了方便,我们直接在HelloWorldLib.cpp里定义函数
C++库导出有两种方式
一、以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;
}
二、以模块定义文件的方式导出
在工程上右键,选择添加-》新建项
然后选择代码-》模块定义文件
在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位的,否则会报错。
2、使用C#调用
接下来我们新建一个C#控制台项目
打开前面C++库生成的目录,将HelloWorldLib.dll复制到C#工程的Debug目录下。也可以不复制,只需在引用dll的时候写上完整路径就行了。这里我是直接复制到Debug目录下
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);
}
}
}
}
}
}
运行结果如下:
C#与C++互操作,很重要的一个地方就是,要注意数据类型的对应。有时还需要加上一些限制,
关于C#与C++数据类型对应
可以参考以下链接:
https://www.cnblogs.com/zjoch/p/5999335.html
大部分硬件厂商提供的SDK都是需要C++来调用的,有了上面的知识,使用C#来调用一些硬件的SDK就比较容易了。只需要使用C++再进行一次封装就行了。
二、C++调用C#库
这里用到是C++/CLI,就是如何用C++在·NET中编程。就是因为有这个东西的存在,C++才能调用C#的库
下面新建一个C#类库CSharpLib
未完。。。。
C#与C++与互操作的更多相关文章
- 没有为 COM 互操作注册程序集 请使用 regasm.exe /tlb 注册该程序集——解决办法
错误现象: 错误 6 没有为 COM 互操作注册程序集“DevExpress.Utils.v13.1, Version=13.1.7.0, Culture=neutral, PublicKeyToke ...
- VS2013中, 无法嵌入互操作类型“……”,请改用适用的接口的解决方法
使用VS2013,在引用COM组件的时候,出现了无法嵌入互操作类型“……”,请改用适用的接口的错误提示. 查阅资料,找到解决方案,记录如下: 选中项目中引入的dll,鼠标右键,选择属性,把“嵌入互操作 ...
- VS2010中,无法嵌入互操作类型“……”,请改用适用的接口的解决方法(转自网络)
最近开始使用VS2010,在引用COM组件的时候,出现了无法嵌入互操作类型“……”,请改用适用的接口的错误提示.查阅资料,找到解决方案,记录如下: 选中项目中引入的dll,鼠标右键,选择属性,把“嵌入 ...
- 与Java互操作
课程内容涵盖了Java互操作性. Javap 类 异常 特质 单例对象 闭包和函数 变化性 Javap javap的是JDK附带的一个工具.不是JRE,这里是有区别的. javap反编译类定义,给你展 ...
- Microsoft.Office.Interop.Excel 程序集引用 ,Microsoft.Office.Interop.Excel.ApplicationClass 无法嵌入互操作类型
using Microsoft.Office.Interop.Excel 添加程序集引用 方法:在引用--程序集--扩展中,添加引用Microsoft.Office.Interop.Excel,此 ...
- ArcEngine 无法嵌入互操作类型
说明: 在.net 4.0中,声明 IPoint point = new PointClass();会出现下面这个错误 错误 2 类型"ESRI.ArcGIS.Geometry.PointC ...
- 错误 24 无法嵌入互操作类型“ESRI.ArcGIS.Geometry.PointClass”。请改用适用的接口。 E:\MyGIS\MyGIS\Form1.cs 78 37 MyGIS
解决办法:选中那个引用,在属性页,将“嵌入互操作”设置为false
- 无法嵌入互操作类型“ESRI.ArcGIS.Carto.RectangleElementClass”。请改用适用的接口。
右键点击应用的程序集 ESRI.ArcGIS.Controls,修改"嵌入互操作类型"的值即可
- 无法嵌入互操作类型“Microsoft.Office.Interop.Excel.ApplicationClass”。请改用适用的接口
解决 把Microsoft.Office.Interop.Excel.DLL的嵌入互操作类型改为ture就可以了
- ArcEngine10.1二次开发错误: 无法嵌入互操作类型,请改用适用的接口
在之前配置ArcEngine.VS2010二次开发程序的时候,遇见"无法嵌入互操作类型,请改用适用的接口"的错误,在网上查了下,下面引用解决方法. 解决方式为在提示错误的引用上面右 ...
随机推荐
- Android 实现系统分享
使用手机上的程序,来分享/发送,比如QQ的“发送到我的电脑”. 1.分享/发送文本内容 Intent shareIntent = new Intent(); shareIntent.setAction ...
- 剑指:和为S的连续正数序列
题目描述 输入一个正数 s,打印出所有和为 s 的连续正数序列(至少含有两个数). 例如输入 15,由于 1+2+3+4+5=4+5+6=7+8=15,所以结果打印出 3 个连续序列 1-5.4-6 ...
- Linux 初识Libevent网络库
初识Libevent libevent是用c写的高并发网络io库,只要有文件描述符,就都可使用libevent. libevent使用回调函数(callback) . 有了libevent,网络编程我 ...
- LaTeX安装与编译中文
首先,感谢博客园团队帮我找回这篇被我误删除的博客.找回方法:发送邮件至"contact@cnblogs.com",然后就可以在工作人员的帮助下找回了.下面介绍LaTeX的安装并使其 ...
- 简单的深度神经网络实现——使用PyTorch
使用的数据集是MNIST,预期可以达到98%左右的准确率. 该神经网络由一个输入层,一个全连接层结构的隐含层和一个输出层构建. 1.配置库和配置参数 import torch import torch ...
- glog的编译和使用
glog是google提供的一个轻量级日志库,有chromium开发经验的人都会发现,它和base库中的日志库非常像,其实base库中的日志库比它更加轻量级.glog在日常开发中的使用非常广泛.这里介 ...
- flask 基础2
一.装饰器的坑 在使用装饰器函数时候,当一个装饰器装饰多个函数的时候,会由于内存地址相同时发生报错,因为装饰的都是一个函数 所以就需要引入 import functools 重新定义每一个函数的名称 ...
- 多线程(四)多线程同步_Critical Section临界区
临界区是指一个小代码段,在代码能够执行前,它必须独占对某些共享资源的访问权.和使用mutex一样,它们都是以原子操作方式来对共享资源进行访问. 临界区又叫关键代码段,与上一篇的mutex互斥体实现的功 ...
- C++ 函数重载、函数模板,类模板
1.函数重载 相同作用域下,有多个函数名相同,但形参列表不同的函数,常用于处理功能相同但数据类型不同的问题 函数重载的规则: 函数名必须相同 函数形参列表必须不同(可以是参数个数不同,或者数据类型不同 ...
- 莫烦TensorFlow_08 tensorboard可视化进阶
import tensorflow as tf import numpy as np import matplotlib.pyplot as plt # # add layer # def add_l ...