• (转)C#调用DLL时参数问题的一点心得


    C#导入DLL时,参数怎么定义是一个比较头痛的问题。特别是指针类型的参数,关于此问题本人有点不成熟的经验。

          以 GetComputerName这个函数为例。

         函数原型如下

    BOOL GetComputerName(
      LPTSTR lpBuffer,
      LPDWORD lpnSize
    );
    

    这个lpBuffer就是下个string型的指针,其实无论是什么类型的指针,对于Windows来说都是一个32位的无符号的整数,也就是一个内在地址,函数之所以使用指针就是要向指针所指向的内存空间写入数据。

    我们用C#调用时也要给它传递一个指针,还要对应一块分配的空间 。

    下面是代码:

    Code

    其实和C++在调用的本质是一样的,都要分配空间并将空间的地址传给函数

    。只是C#是运行在托管环境,所以对空间的分配的数据的读取都要特殊处理。

    还是更简单的方法,就是用StringBuilder。本质上还是和上面一样的,只不过是C#替你做了得多工作。

    Code

    对于结构体也是一样,以

    BOOL GetVersionEx(
      LPOSVERSIONINFO lpVersionInformation
    );

    typedef struct _OSVERSIONINFO{
      DWORD dwOSVersionInfoSize;
      DWORD dwMajorVersion;
      DWORD dwMinorVersion;
      DWORD dwBuildNumber;
      DWORD dwPlatformId;
      TCHAR szCSDVersion[128];
    } OSVERSIONINFO;

    为例说明一下。

    参数是一个结构体的指针,函数会填充结构体的各个字段,其实就是向这块内存空间的不同位置写入不同的数据。

    Code

    根据结构体的定义有5个DWORD和一个128位的CHAR数组,所以要给这个结构体分配148位空间。结构体的第一个字段这是个结构体的大小,我们用 Marshal.WriteInt32(pv, 148);写入结构体的大小(这也是Windows API在使用结构体的一个特点,就是大部分结构体在传给函数填充前要指定其大小。),然后执行函数。

    如果函数正常返回,就可以根据结构体的定义从相应的位置读出数据。

    Marshal.ReadInt32(pv, 4);读取第二个整数也就是MajorVersion,Marshal.ReadInt32(pv, 12);第四个是BuildNumber,Marshal.PtrToStringAnsi((IntPtr)(pv.ToInt32() + 20));五个整数之后是CSDVersion。

    写上面的内容只是想分析一下机理,如果学习过汇编就很好理解了。

    下面是通常的做法:

    Code
  • 相关阅读:
    攻防世界web新手区前六关
    JS-数组基础知识3
    CSRF攻击的原理和spring security对CSRF攻击的解决方法
    Java开发微信公众号
    内部类
    Java Web整合开发(30) -- Spring的ORM模块
    win10安装mysql
    jquery 事件冒泡的介绍以及如何阻止事件冒泡
    jquery中attr和prop的区别介绍
    jQuery 层次选择器
  • 原文地址:https://www.cnblogs.com/fjchenqian/p/1445258.html
Copyright © 2020-2023  润新知