C# 调用DLL 常用如下方式来引用库中的函数 (当然还有很多其它的实现方式,这里就不多讲),本文主要讲要使用中遇到的常见问题:

一,库的引用 加上 :using System.Runtime.InteropServices;

[DllImport("XXX")]
static extern void UbtServoGroupInit(ref UbtUartServo Sgroup, pwrite write, pread read);

上面有注意的有两点:

1,XXX 可以是DLL库名不带.dll 后辍,也可以是全名

2,函数前的 static extern 是一定要加的

二,DLL头文件实现

C#中没有.H文件,如果DLL中用到了一些头文件(里面有结构体等自定义类型),需在要C#环境中从新定义,这里会有各种变量的对应转换关系:

/****** 以下内容引自于https://www.cnblogs.com/zhaoxinshanwei/p/4008627.html  ***********/

//c++:HANDLE(void   *)       ----   c#:System.IntPtr        
//c++:Byte(unsigned   char)   ----    c#:System.Byte         
//c++:SHORT(short)           ----    c#:System.Int16         
//c++:WORD(unsigned short) ---  c#:System.UInt16        
//c++:INT(int)                 ----    c#:System.Int16        
//c++:INT(int)                  ----    c#:System.Int32         
//c++:UINT(unsigned int)  ----    c#:System.UInt16        
//c++:UINT(unsigned int) ----    c#:System.UInt32        
//c++:LONG(long)             ----    c#:System.Int32         
//c++:ULONG(unsigned long) --   c#:System.UInt32         
//c++:DWORD(unsigned long) --  c#:System.UInt32         
//c++:DECIMAL         ----    c#:System.Decimal         
//c++:BOOL(long)        ----    c#:System.Boolean         
//c++:CHAR(char)       ----    c#:System.Char  
       
//c++:LPSTR(char *)      ----    c#:System.String         
//c++:LPWSTR(wchar_t *)  ----    c#:System.String         
//c++:LPCSTR(const char *)  ---   c#:System.String         
//c++:LPCWSTR(const wchar_t *) ---   c#:System.String

//c++:PCAHR(char *)   ----   c#:System.String     
    
//c++:BSTR       ----    c#:System.String       
  
//c++:FLOAT(float)      ----    c#:System.Single         
//c++:DOUBLE(double)    ----    c#:System.Double         
//c++:VARIANT           ----    c#:System.Object         
//c++:PBYTE(byte   *)   ----    c#:System.Byte[]         
//c++:BSTR      ----    c#:StringBuilder    
    
//c++:LPCTSTR   ----    c#:StringBuilder  
      
//c++:LPCTSTR   ----    c#:string

//c++:LPTSTR    ----    
c#:[MarshalAs(UnmanagedType.LPTStr)] string  
       
//c++:LPTSTR     ----    c#:StringBuilder 
     
//c++:LPCWSTR   ----    c#:IntPtr

//c++:BOOL      ----    c#:bool

//c++:HMODULE   ----    c#:IntPtr        
    
//c++:HINSTANCE ----    c#:IntPtr        
 
//c++:结构体    ----    c#:public struct 结构体{}; 
        
//c++:结构体 **变量名   ----    c#:out 变量名

//C#中提前申明一个结构体实例化后的变量名      
 
//c++:结构体 &变量名    ----    c#:ref 结构体 变量名                 
//c++:WORD      ----    c#:ushort

//c++:DWORD     ----    c#:uint

//c++:DWORD     ----    c#:int

//c++:UCHAR     ----    c#:int

//c++:UCHAR     ----    c#:byte

//c++:UCHAR*    ----    c#:string      
  
//c++:UCHAR*    ----    c#:IntPtr

//c++:GUID      ----    c#:Guid

//c++:Handle    ----    c#:IntPtr

//c++:HWND      ----    c#:IntPtr

//c++:DWORD     ----    c#:int

//c++:COLORREF  ----    c#:uint       
 
//c++:unsigned char     ----    c#:byte      
  
//c++:unsigned char *   ----    c#:ref byte       
 
//c++:unsigned char *   ----    
c#:[MarshalAs(UnmanagedType.LPArray)] byte[]        
//c++:unsigned char *   ----    
c#:[MarshalAs(UnmanagedType.LPArray)] Intptr        
//c++:unsigned char &   ----    c#:ref byte  
      
//c++:unsigned char 变量名      ----    c#:byte 变量名        
//c++:unsigned short 变量名    ----   c#:ushort 变量名        
//c++:unsigned int 变量名       ----    c#:uint 变量名        
//c++:unsigned long 变量名     ----    c#:ulong 变量名        
//c++:char 变量名       ----    c#:byte 变量名 
  
//c++中一个字符用一个字节表示,c#中一个字符用两个字节表示

//c++:char 数组名[数组大小]     ----    
c#:MarshalAs(UnmanagedType.ByValTStr, SizeConst = 数组大小)]        
   
//c++:char *            ----    c#:string       
 
//c++:char *            ----    c#:StringBuilder
    
//c++:char *变量名      ----    c#:ref string 变量名        
//c++:char *输入变量名  ----    c#:string 输入变量名        
//c++:char *输出变量名  ----    
c#:[MarshalAs(UnmanagedType.LPStr)] StringBuilder 输出变量名      
  
//c++:char **           ----    c#:string     
   
//c++:char **变量名     ----    c#:ref string 变量名        
//c++:const char *      ----    c#:string   
     
//c++:char[]            ----    c#:string     
   
//c++:char 变量名[数组大小]     ----    
c#:[MarshalAs(UnmanagedType.ByValTStr,SizeConst=数组大小)] public string 变量名;

//c++:struct 结构体名 *变量名   ----    c#:ref 结构体名 变量名

//c++:委托 变量名   ----    c#:委托 变量名    
    
//c++:int       ----    c#:int

//c++:int       ----    c#:ref int

//c++:int &     ----    c#:ref int      
  
//c++:int *     ----    c#:ref int

//C#中调用前需定义int 变量名 = 0;      
  
//c++:*int      ----    c#:IntPtr

//c++:int32 PIPTR *     ----    c#:int32[]   
     
//c++:float PIPTR *     ----    c#:float[]       
         
//c++:double** 数组名          ----    c#:ref double 数组名

//c++:double*[] 数组名          ----    c#:ref double 数组名

//c++:long          ----    c#:int

//c++:ulong         ----    c#:int           
     
//c++:UINT8 *       ----    c#:ref byte      
 
//C#中调用前需定义byte 变量名 = new byte();                
//c++:handle    ----    c#:IntPtr

//c++:hwnd      ----    c#:IntPtr         
               
//c++:void *    ----    c#:IntPtr          
      
//c++:void * user_obj_param    ----    
c#:IntPtr user_obj_param

//c++:void * 对象名称    ----    
c#:([MarshalAs(UnmanagedType.AsAny)]Object 对象名称

//c++:char,INT8,SBYTE,CHAR  --  c#:System.SByte          
//c++:short, short int, INT16, SHORT                        ----    c#:System.Int16        
  
//c++:int, long, long int,INT32, LONG32, BOOL , INT  ----    c#:System.Int32       
   
//c++:__int64,INT64,LONGLONG                           --  c#:System.Int64

//c++:unsigned char, UINT8, UCHAR , BYTE               ----    c#:System.Byte

//c++:unsigned short, UINT16, USHORT, WORD, ATOM, WCHAR , __wchar_t  ----  c#:System.UInt16

//c++:unsigned, unsigned int, UINT32, ULONG32, DWORD32, ULONG, DWORD, UINT     
 ----    c#:System.UInt32

//c++:unsigned __int64, UINT64, DWORDLONG, ULONGLONG                            
----    c#:System.UInt64

//c++:float,FLOAT    --  c#:System.Single

//c++:double, long double, DOUBLE                          ----    c#:System.Double

//Win32 Types        ----  CLR Type       
           
//Struct需要在C#里重新定义一个Struct    
    
//CallBack回调函数需要封装在一个委托里,delegate static extern int FunCallBack(string str);       
 
//unsigned char** ppImage替换成IntPtr ppImage        
//int& nWidth替换成ref int nWidth       
 
//int*, int&, 则都可用 ref int 对应      
  
//双针指类型参数,可以用 ref IntPtr       
 
//函数指针使用c++: typedef double (*fun_type1)(double); 对应 c#:public delegate double  fun_type1(double);

//char* 的操作c++: char*; 对应 c#:StringBuilder;        
//c#中使用指针:在需要使用指针的地方 加 unsafe

//unsigned   char对应public   byte

/****** 以上内容引自于https://www.cnblogs.com/zhaoxinshanwei/p/4008627.html  ***********/

根据上面的对应关系重新实现结构体或自定义类型,如:

public enum Status
    {
        Servo_OK = 0,
        Servo_TimeOut,
        Servo_ParErr,
        Servo_LibInitErr,
        Servo_TemperatureLow,
        Servo_TemperatureHight,
        Servo_VoltageLow,
        Servo_VoltageHight,
        Servo_CurrentOver,
        Servo_TorqueOver,
        Servo_FuseErr,
        Servo_PwmErr,
        Servo_CmdFail,
        Servo_ReciveErr,
    };

public struct UbtUartServo
    {
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 64)]
         string RxBuf;

//1.定义回调函数指针.
        public pwrite write;
        public pread read;
    };

三,C#中如何实现类似于函数指针功能 (delegate)委托:

C/C++中的 typedef void (*pwrite)Byte buf, Byte size);

在C#可对应为:delegate  void pwrite(ref Byte buf, Byte size);

如 在 C 中

typedef struct 
{
    uint8_t RxBuf[64];
    void   (*pwrite)(uint8_t *buf,uint8_t size)     C51;
    uint8_t   (*pread)(uint8_t *buf,uint8_t size,uint8_t mstimeout)  C51;
} UbtUartServo;

在C#的实现为:

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate  void pwrite(ref Byte buf, Byte size);

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate  Byte pread(ref Byte buf, Byte size, Byte mstimeout);

public struct UbtUartServo
    {
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 64)]
         string RxBuf;

//1.定义回调函数指针.
        public pwrite write;
        public pread read;
    };

其中上面的,Cdecl说明DLL是C格式!

C++或其它可能要用不同的格式(前三个用的多一些,不过不行可以试度其它):

namespace System.Runtime.InteropServices
{
    [Serializable]
    [ComVisible(true)]
    public enum CallingConvention
    {
        Winapi = 1,
        Cdecl = 2,
        StdCall = 3,
        ThisCall = 4,
        FastCall = 5,
    }
}

不然有可能会出现 如下错误:

The value of ESP was not properly saved across a function call. This is usually a result of calling a function declared with one calling convention with a function pointer declared with a different calling convention.

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate  void pwrite(ref Byte buf, Byte size);

四,给个例子:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace CshareServoTest
{

public enum Status
    {
        Servo_OK = 0,
        Servo_TimeOut,
        Servo_ParErr,
        Servo_LibInitErr,
        Servo_TemperatureLow,
        Servo_TemperatureHight,
        Servo_VoltageLow,
        Servo_VoltageHight,
        Servo_CurrentOver,
        Servo_TorqueOver,
        Servo_FuseErr,
        Servo_PwmErr,
        Servo_CmdFail,
        Servo_ReciveErr,
    };

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate  void pwrite(ref Byte buf, Byte size);

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate  Byte pread(ref Byte buf, Byte size, Byte mstimeout);

public struct UbtUartServo
    {
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 64)]
         string RxBuf;

//1.定义回调函数指针.
        public pwrite write;
        public pread read;
    };

public partial class Form1 : Form
    {
        [DllImport("ServoLibDll")]
        static extern void UbtServoGroupInit(ref UbtUartServo Sgroup, pwrite write, pread read);

[DllImport("ServoLibDll")]
        static extern Status UbtServoMove(ref UbtUartServo Sgroup, Byte ServoID, Byte TargetAngle, Int16 RunTime, UInt16 KeepTime, Byte BackTimeOut);

public UbtUartServo Test;

public Form1()
        {
            InitializeComponent();
            Test = new UbtUartServo();

//Test.write = new pwrite(write);
            //Test.read = new pread(read);

UbtServoGroupInit(ref Test, new pwrite(this.write), new pread(this.read));
        }

private void TestButton_Click(object sender, EventArgs e)
        {
            UbtServoMove(ref Test, 0, 120, 25, 500, 50);
        }

public void write(ref Byte buf, Byte size)
        {
            ServoTest.Text = "write";
            return;
        }

public Byte read(ref Byte buf, Byte size, Byte mstimeout)
        {
            ServoTest.Text = "read";
            return 1;
        }

}
}

C#调用C/C++ DLL的相关说明相关推荐

  1. java 项目加载dll文件,Java动态调用链接库dll文件

    Java动态调用链接库dll文件 Java动态调用链接库dll文件 因为项目需要,用到了java调用dll文件中的方法.写成dll文件大多是用C语言弄成的,对于我这个纯粹干java的人员来说,着实让我 ...

  2. 32位python-64位Python调用32位DLL方法(一)

    摘要: 本文介绍了一种简单快捷的方法,实现在64位Python调用32位DLL. 引言: Python发展到现在,3.x的普及率已经相当高,而且随着硬件(内存的不断增大)以及操作系统(64bit wi ...

  3. C#.net调用动态库dll注意事项

    C#.net调用动态库dll注意事项 本文主要描述作者在工作中所遇到技术难点及问题,最后提出相关的解决方案 场景 公司有个业务需求涉及到ID读卡器,构建成在ID卡读卡器一刷ID卡,系统就自动显示其ID ...

  4. 在C#里调用C++的dll时需要注意的一些问题转

    原文来自:http://hi.baidu.com/cityhacker/blog/item/419ed50af30a9e1595ca6b9d.html 2009-11-19 12:21 在c#里调用C ...

  5. 64位进程调用32位dll的解决方法 / 程序64位化带来的问题和思考

    最近做在Windows XP X64,VS2005环境下做32位程序编译为64位程序的工作,遇到了一些64位编程中可能遇到的问题:如内联汇编(解决方法改为C/C++代码),long类型的变化,最关键的 ...

  6. 在C#调用C++的DLL简析(二)—— 生成托管dll

    写操作之前,还是扼要的说一下托管与非托管C++的区别好了,其实我也并没有深入了解过托管C++的特点所在,其最大的特征就是可以由系统来调试回收相关的代码资源,跟C#的特性一样,只是编程风格跟C++类似而 ...

  7. c 调用c语言dll数组,C#调用C类型dll入参为struct的问题详解

    前言 C# 可以通过 DllImport 的方式引用 C 类型的 dll.但很多 dll 的参数不会是简单的基础类型,而是结构体 struct .因此就需要在 C# 端定义同样的结构体类型,才能实现调 ...

  8. 64位进程调用32位dll的解决方法

    最近做在Windows XP X64,VS2005环境下做32位程序编译为64位程序的工作,遇到了一些64位编程中可能遇到的问题:如内联汇编(解决方法改为C/C++代码),long类型的变化,最关键的 ...

  9. 一种实现 Java调用C++的DLL的方法

    http://topic.csdn.net/t/20020917/14/1030420.html 感谢大家支持,这几天我在IBM中文及SUN官方网站中找了不少关于JNI的文章,受益非浅,希望     ...

最新文章

  1. 五种常见的 PHP 设计模式
  2. xp主机用VMware9和10安装Ubuntu12.04后无法进入图像界面
  3. linux mate eth0已下线,Linux Mint 18 Cinnamon 版和 MATE 版已经可以下载了
  4. [转载] python2.7中模块学习- textwrap 文本包装和填充
  5. https阿里云证书购买与apache环境配置
  6. 《流畅的Python第二版》读书笔记——字典和集合
  7. ToStringBuilder用法
  8. 计算二维离散随机变量的联合概率分布
  9. 分布式机器学习联邦学习论文阅读笔记(持续更新)
  10. 服务注册与发现-Spring Cloud Netflix-Eureka
  11. 三角形,斜线,表头css实现方法
  12. 孟:解密获取网站访客QQ号码的方法及原理
  13. OSI与TCP/IP各层的结构与功能,都有哪些协议
  14. 在线视频转音频怎么弄?这几个软件建议你们收藏
  15. WPF(七) Prism框架基本特性
  16. 2021年中国Wifi智能电源插座市场趋势报告、技术动态创新及2027年市场预测
  17. Perpendicular
  18. 加权GN算法的Java实现
  19. 打开软件字体显示太小解决办法
  20. 商务工作汇报年终总结PPT模板

热门文章

  1. 持续集成之jenkins基础
  2. HTML 统一资源定位器(Uniform Resource Locators)URL
  3. 『TCP/IP详解——卷一:协议』读书笔记——03
  4. SQLite的事务和锁
  5. Log Cauchy分布的一个Hierarchical模型:LC=Gamma+Gamma+Unif
  6. C语言scanf函数的返回值、scanf函数的安全版、在while条件中使用scanf函数
  7. 不同浏览器前端调试查看返回页面的json数据
  8. 资源文件相关学习总结
  9. Asp.Net 将HTML中通过dom-to-image.js标签div内的内容转化为图片保存到本地
  10. Git--团队开发必备神器