当前位置: 首页 > 编程日记 > 正文

C#中调用Windows API的要点

在.Net Framework SDK文档中,关于调用Windows API的指示比较零散,并且其中稍全面一点的是针对Visual Basic .net讲述的。本文将C#中调用API的要点汇集如下,希望给未在C#中使用过API的朋友一点帮助。另外如果安装了Visual Studio .net的话,在C:/Program Files/Microsoft Visual Studio .NET/FrameworkSDK/Samples/Technologies/Interop/PlatformInvoke/WinAPIs/CS目录下有大量的调用API的例子。
一、调用格式
using System.Runtime.InteropServices; //引用此名称空间,简化后面的代码
...
//使用DllImportAttribute特性来引入api函数,注意声明的是空方法,即方法体为空。
[DllImport("user32.dll")]
public static extern ReturnType FunctionName(type arg1,type arg2,...);
//调用时与调用其他方法并无区别

可以使用字段进一步说明特性,用逗号隔开,如:
 [ DllImport( "kernel32", EntryPoint="GetVersionEx" )]
 DllImportAttribute特性的公共字段如下:
 1、CallingConvention 指示向非托管实现传递方法参数时所用的 CallingConvention 值。
  CallingConvention.Cdecl : 调用方清理堆栈。它使您能够调用具有 varargs 的函数。
  CallingConvention.StdCall : 被调用方清理堆栈。它是从托管代码调用非托管函数的默认约定。
 2、CharSet 控制调用函数的名称版本及指示如何向方法封送 String 参数。
  此字段被设置为 CharSet 值之一。如果 CharSet 字段设置为 Unicode,则所有字符串参数在传递到非托管实现之前都转换成 Unicode 字符。这还导致向 DLL EntryPoint 的名称中追加字母“W”。如果此字段设置为 Ansi,则字符串将转换成 ANSI 字符串,同时向 DLL EntryPoint 的名称中追加字母“A”。大多数 Win32 API 使用这种追加“W”或“A”的约定。如果 CharSet 设置为 Auto,则这种转换就是与平台有关的(在 Windows NT 上为 Unicode,在 Windows 98 上为 Ansi)。CharSet 的默认值为 Ansi。CharSet 字段也用于确定将从指定的 DLL 导入哪个版本的函数。CharSet.Ansi 和 CharSet.Unicode 的名称匹配规则大不相同。对于 Ansi 来说,如果将 EntryPoint 设置为“MyMethod”且它存在的话,则返回“MyMethod”。如果 DLL 中没有“MyMethod”,但存在“MyMethodA”,则返回“MyMethodA”。对于 Unicode 来说则正好相反。如果将 EntryPoint 设置为“MyMethod”且它存在的话,则返回“MyMethodW”。如果 DLL 中不存在“MyMethodW”,但存在“MyMethod”,则返回“MyMethod”。如果使用的是 Auto,则匹配规则与平台有关(在 Windows NT 上为 Unicode,在 Windows 98 上为 Ansi)。如果 ExactSpelling 设置为 true,则只有当 DLL 中存在“MyMethod”时才返回“MyMethod”。

3、EntryPoint 指示要调用的 DLL 入口点的名称或序号。
  如果你的方法名不想与api函数同名的话,一定要指定此参数,例如:
 [DllImport("user32.dll",CharSet="CharSet.Auto",EntryPoint="MessageBox")]
 public static extern int MsgBox(IntPtr hWnd,string txt,string caption, int type);

4、ExactSpelling 指示是否应修改非托管 DLL 中的入口点的名称,以与 CharSet 字段中指定的 CharSet 值相对应。如果为 true,则当 DllImportAttribute.CharSet 字段设置为 CharSet 的 Ansi 值时,向方法名称中追加字母 A,当 DllImportAttribute.CharSet 字段设置为 CharSet 的 Unicode 值时,向方法的名称中追加字母 W。此字段的默认值是 false。
 5、PreserveSig 指示托管方法签名不应转换成返回 HRESULT、并且可能有一个对应于返回值的附加 [out, retval] 参数的非托管签名。
 6、SetLastError 指示被调用方在从属性化方法返回之前将调用 Win32 API SetLastError。 true 指示调用方将调用 SetLastError,默认为 false。运行时封送拆收器将调用 GetLastError 并缓存返回的值,以防其被其他 API 调用重写。用户可通过调用 GetLastWin32Error 来检索错误代码。

二、参数类型:
 1、数值型直接用对应的就可。(DWORD -> int , WORD -> Int16)
 2、API中字符串指针类型 -> .net中string
 3、API中句柄 (dWord)  -> .net中IntPtr
 4、API中结构   -> .net中结构或者类。注意这种情况下,要先用StructLayout特性限定声明结构或类
公共语言运行库利用StructLayoutAttribute控制类或结构的数据字段在托管内存中的物理布局,即类或结构需要按某种方式排列。如果要将类传递给需要指定布局的非托管代码,则显式控制类布局是重要的。它的构造函数中用LayoutKind值初始化 StructLayoutAttribute 类的新实例。 LayoutKind.Sequential 用于强制将成员按其出现的顺序进行顺序布局。
 LayoutKind.Explicit 用于控制每个数据成员的精确位置。利用 Explicit, 每个成员必须使用 FieldOffsetAttribute 指示此字段在类型中的位置。如:
 [StructLayout(LayoutKind.Explicit, Size=16, CharSet=CharSet.Ansi)]
 public class MySystemTime
 {
    [FieldOffset(0)]public ushort wYear;
    [FieldOffset(2)]public ushort wMonth;
    [FieldOffset(4)]public ushort wDayOfWeek;
    [FieldOffset(6)]public ushort wDay;
    [FieldOffset(8)]public ushort wHour;
    [FieldOffset(10)]public ushort wMinute;
    [FieldOffset(12)]public ushort wSecond;
    [FieldOffset(14)]public ushort wMilliseconds;
 }
 下面是针对API中OSVERSIONINFO结构,在.net中定义对应类或结构的例子:
/**********************************************
* API中定义原结构声明
* OSVERSIONINFOA STRUCT
*  dwOSVersionInfoSize   DWORD      ?
*  dwMajorVersion        DWORD      ?
*  dwMinorVersion        DWORD      ?
*  dwBuildNumber         DWORD      ?
*  dwPlatformId          DWORD      ?
*  szCSDVersion          BYTE 128 dup (?)
* OSVERSIONINFOA ENDS
*
* OSVERSIONINFO  equ  <OSVERSIONINFOA>
*********************************************/

//.net中声明为类
[ StructLayout( LayoutKind.Sequential )]  
public class OSVersionInfo
{   
    public int OSVersionInfoSize;
    public int majorVersion;
    public int minorVersion;
    public int buildNumber;
    public int platformId;

[ MarshalAs( UnmanagedType.ByValTStr, SizeConst=128 )]   
    public String versionString;
}
//或者
//.net中声明为结构
[ StructLayout( LayoutKind.Sequential )] 
public struct OSVersionInfo2
{
    public int OSVersionInfoSize;
    public int majorVersion;
    public int minorVersion;
    public int buildNumber;
    public int platformId;

[ MarshalAs( UnmanagedType.ByValTStr, SizeConst=128 )]   
    public String versionString;
}

此例中用到MashalAs特性,它用于描述字段、方法或参数的封送处理格式。用它作为参数前缀并指定目标需要的数据类型。例如,以下代码将两个参数作为数据类型长指针封送给 Windows API 函数的字符串 (LPStr):
    [MarshalAs(UnmanagedType.LPStr)]
 String existingfile;
    [MarshalAs(UnmanagedType.LPStr)]
 String newfile;

注意结构作为参数时候,一般前面要加上ref修饰符,否则会出现错误:对象的引用没有指定对象的实例。
 [ DllImport( "kernel32", EntryPoint="GetVersionEx" )]
 public static extern bool GetVersionEx2( ref OSVersionInfo2 osvi );

三、如何保证使用托管对象的平台调用成功?
    如果在调用平台 invoke 后的任何位置都未引用托管对象,则垃圾回收器可能将完成该托管对象。这将释放资源并使句柄无效,从而导致平台invoke 调用失败。用 HandleRef 包装句柄可保证在平台 invoke 调用完成前,不对托管对象进行垃圾回收。
    例如下面:
        FileStream fs = new FileStream( "a.txt", FileMode.Open );
        StringBuilder buffer = new StringBuilder( 5 );
        int read = 0;
        ReadFile(fs.Handle, buffer, 5, out read, 0 ); //调用Win API中的ReadFile函数
 由于fs是托管对象,所以有可能在平台调用还未完成时候被垃圾回收站回收。将文件流的句柄用HandleRef包装后,就能避免被垃圾站回收:
 [ DllImport( "Kernel32.dll" )]
 public static extern bool ReadFile(
  HandleRef hndRef,
  StringBuilder buffer,
  int numberOfBytesToRead,
  out int numberOfBytesRead,
  ref Overlapped flag );
 ......
 ......
        FileStream fs = new FileStream( "HandleRef.txt", FileMode.Open );
        HandleRef hr = new HandleRef( fs, fs.Handle );
        StringBuilder buffer = new StringBuilder( 5 );
        int read = 0;
        // platform invoke will hold reference to HandleRef until call ends
        ReadFile( hr, buffer, 5, out read, 0 );

[作者hahahawk 主页http://www.qiong.net/ 邮箱mailto:cangqiong@sina.com]

相关文章:

线上直播丨Hinton等6位图灵奖得主、百余位顶级学者邀你群聊AI

Geoffrey Hinton等6位图灵奖得主亲临&#xff0c;百余位顶级学者邀请你加入群聊「2020北京智源大会」&#xff0c;深入系统探讨「人工智能的下一个十年」。自2009年深度学习崛起以来&#xff0c;第三波人工智能浪潮席卷全球&#xff0c;推动了新一波技术革命。在这波澜壮阔的11…

ServerSocket

ServerScoket 这个类用于与 Socket 进行通信。 在实例化ServerSocket 的时候&#xff0c;服务器相当于已经开始了&#xff0c;但是还需要通过socket来accept &#xff08;socket serverSocket.accept()&#xff09;以使服务器选择性与某一Client进行连接。如果有指定了允许连接…

NDK开发 - C/C++ 访问 Java 变量和方法

上一篇有提到 JNI 访问引用数组&#xff0c;涉及了 C/C 访问 Java 实例的方法和变量。虽然在之前的开发中&#xff0c;并没有用到 C/C 范围 Java 层数据&#xff0c;但是这部分内容还是很有用的。传送门&#xff1a;NDK开发 - C/C 访问 Java 变量和方法 C/C 访问 Java 层的方法…

在C#中应用哈希表(Hashtable)

一,哈希表(Hashtable)简述 在.NET Framework中&#xff0c;Hashtable是System.Collections命名空间提供的一个容器&#xff0c;用于处理和表现类似key/value的键值对&#xff0c;其中key通常可用来快速查找&#xff0c;同时key是区分大小写&#xff1b;value用于存储对应于key的…

俄罗斯自研Elbrus CPU参数曝光,CEO年近九旬仍未退休

导语&#xff1a;俄罗斯自研 CPU 参数最近曝光&#xff0c;虽然比起主流产品仍存在较大差距&#xff0c;但是这也是俄罗斯在自研道路上的一大进展。虽说自研 CPU 并非易事&#xff0c;但为了避免被美国牵制&#xff0c;很多国家都在这方面投入了巨大的人力与资金。战斗民族俄罗…

停止Password Manager Agent服务导致应用程序启动缓慢

在一个实施环境中&#xff0c;部署了Password Manager用来实现单点登录功能&#xff0c;但是由于Password Manager的提示基本都是以英文为主&#xff0c;而且配置也比较麻烦&#xff0c;普通用户看见会比较影响用户体验&#xff0c;所以用户决定暂时关闭Password Manager功能&a…

web 前端常用组件【06】Upload 控件

因为有万恶的IE存在&#xff0c;所以当Web项目初始化并进入开发阶段时。 如果是项目经理&#xff0c;需要知道客户将会用什么浏览器来访问系统。 明确知道限定浏览器的情况下&#xff0c;你才能从容的让手下的封装必要的前端组件。 本篇文章试图从常见的上传方式和组件进行分析…

性能超越最新序列推荐模型,华为诺亚方舟提出记忆增强的图神经网络

作者 | Chen Ma, Liheng Ma等译者 | Rachel出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09;用户-商品交互的时间顺序可以揭示出推荐系统中用户行为随时间演进的序列性特征。用户与之交互的商品可能受到用户曾经接触的商品的影响。但是&#xff0c;用户和商…

ASP.net 中的页面继承实现和通用页面的工厂模式的实现

最近用.Net做web项目的时候遇到了一些问题&#xff0c;就是很多的页面的处理一样的&#xff0c;不一样的就是我们写的存储过程不同&#xff0c;为了考虑代码的重复利用和可维护性和可 扩展性&#xff0c;于是写了一个对于单据页面的工厂模式&#xff0c;采用界面的继承技术&…

5502的时钟组

5502有四个时钟组&#xff0c;分别为&#xff1a; C55x Subsystem Clock GroupFast Peripherals Clock GroupSlow Peripherals Clock GroupExternal Memory Interface Clock Group1、C55x Subsystem Clock Group该时钟组包括C55X CPU core、内存&#xff08;DARAM和ROM&#xf…

遇到的浏览器兼容问题及应对方法

前言&#xff1a; 上周天的时候有个学长找我帮忙做三张页面&#xff0c;因为没有数据交换之类的&#xff0c;只是单纯的前端页面&#xff0c;想着好久没做东西&#xff0c; 看书都看烦了&#xff0c;所以就接了也当是练手。之前因为没有系统的看书&#xff0c;所以其实很多问题…

浅谈权限设计(来自深空老大)

2019独角兽企业重金招聘Python工程师标准>>> By 深空, 2009-09-13 21:45:07 PHPChina的专家版在谈权限设计&#xff0c;苦于没有权限回帖&#xff0c;特发此博文谈谈简单的权限设计。讨论在这里。 最简单的权限验证&#xff0c;应该是登录态的验证&#xff0c;如果登…

5年Python功力,总结了10个开发技巧

作者 | 写代码的明哥来源 |Python编程时光&#xff08;ID: Cool-Python&#xff09;如何在运行状态查看源代码&#xff1f;查看函数的源代码&#xff0c;我们通常会使用 IDE 来完成。比如在 PyCharm 中&#xff0c;你可以 Ctrl 鼠标点击 进入函数的源代码。那如果没有 IDE 呢&…

怎样给目录加权限0777

# chmod -R 0777 /var/www/html/子目录

php学习,一个简单的Calendar(2) 一个简单的活动页面

有了前面的基础&#xff0c;后面就是将页面展示出来。 预览图如下&#xff1a;1号和31号分别有活动&#xff0c;会一并显示出来 这里需要搞定几个问题&#xff0c;一个就是数据库的连接&#xff0c;我们用\sys\class\class.db_connect.inc.php <?php /* * 数据库操作&#…

涨见识了,在终端执行 Python 代码的 6 种方式

作者 | BRETT CANNON译者 | 豌豆花下猫Python猫为了我们推出的 VS Code 的 Python 插件[1]&#xff0c;我写了一个简单的脚本来生成变更日志[2]&#xff08;类似于Towncrier[3]&#xff0c;但简单些&#xff0c;支持 Markdown&#xff0c;符合我们的需求&#xff09;。在发布过…

ASP.NET中DataGrid鼠标经过感知以及点击行弹出窗口

选择自 xujh 的 Blog 作者Blog&#xff1a;http://blog.csdn.net/xujh/ 很多人说很难&#xff0c;其实就这几行代码。只要在DataGrid1的ItemDataBound中写入下代码即可 private void DataGrid1_ItemDataBound(object sender, System.Web.UI.WebControls.DataGridItemEventAr…

python中的module

Python中的Module是比较重要的概念。常见的情况是&#xff0c;事先写好一个.py文件&#xff0c;在另一个文件中需要import时&#xff0c;将事先写好的.py文件拷贝到当前目录&#xff0c;或者是在sys.path中增加事先写好的.py文件所在的目录&#xff0c;然后import。这样的做法&…

找子串替换(kmp)poj1572

题目链接&#xff1a;http://poj.org/problem?id1572 输入数据时要注意&#xff0c;这里是string型 用getline(cin,origin[i]); #include <string> #include <iostream> #include <algorithm> #include <stdio.h>using namespace std;const int maxn …

dll的概念、dll导出类(转)

1、 DLL的概念DLL(Dynamic Linkable Library)&#xff0c;动态链接库&#xff0c;可以向程序提供一些函数、变量或类。这些可以直接拿来使用。静态链接库与动态链接库的区别&#xff1a;&#xff08;1&#xff09;静态链接库与动态链接库都是共享代码的方式。静态链接库把最后的…

墨奇科技汤林鹏:如何用 AI 技术颠覆指纹识别?

受访者 | 墨奇科技联合创始人& CTO 汤林鹏 记者 | Aholiab&#xff0c;编辑 | Carol 出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09; 随着深度学习等AI技术的成熟&#xff0c;生物识别成为了关注度较高的领域&#xff0c;指纹、人脸、虹膜等识别技术…

ASP.Net ViewState的实现

选择自 timmy3310 的 Blog ViewState是.Net中提出的状态保存的一种新途径&#xff08;实际上也是老瓶装新酒&#xff09;&#xff1b;我们知道&#xff0c;传统的Web程序保存状态的方式有这样几种&#xff1a;1、Application 这是Web应用程序生命期中的全局保存区&#xff0c…

【51CTO学院三周年】遇到

作为一名二流学校的大学生&#xff0c;因为学校的一门嵌入式Linux应用程序开发而喜欢上了嵌入式&#xff0c;但是学校却是只上了一学期的课&#xff0c;无奈只能自己找教程继续学习。在3个月前&#xff0c;无意中找到了朱有鹏老师的嵌入式学习基础视频&#xff0c;通过老师视频…

ASP.NET图象处理详解

作者&#xff1a;未知 请与本人联系在使用ASP的时候&#xff0c;我们时常要借助第三方控件来实现一些图象功能。而现在&#xff0c;ASP.NET的推出&#xff0c;我们已经没有必要再使用第三方控件来实现&#xff0c;因为ASP.NET 已经具有强大的功能来实现一些图象处理。现在&…

IT工作者,你们的爱情是这样的吗?

今天在博客里看到了这篇文章&#xff0c;看完这个视频我随笔写了点自己的感受和看法&#xff0c; 视频链接在下方&#xff1a; http://leidu.blog.51cto.com/3245712/622534 很感谢90 男孩提供&#xff0c;建议IT人员看一下&#xff0c;看完写一下你们的感受吧&#xff01;…

平头哥玄铁处理器Linux新版本,5大亮点速览

来源 | 芯片开放社区为了便于 CPU 评估&#xff0c;系统集成&#xff0c;快速上手玄铁处理器 Linux 操作系统&#xff0c;平头哥更新了玄铁处理器 linux 版本&#xff0c;结合 gitlab 开源 CI/CD 系统&#xff0c;对已发布到开源社区的玄铁架构 CPU 相关的生态软件形成持续保障…

费用保险单,如何失焦时自动补零

费用&#xff0c;如何失焦时自动补零转载于:https://www.cnblogs.com/maojiayan/p/5606247.html

腾讯AI种番茄双丰收:参赛AI全胜专家,辽宁试点净利增千元

6月9日&#xff0c;腾讯宣布了两项AI农业领域进展。在研究侧&#xff0c;腾讯 AI Lab 与荷兰瓦赫宁根大学&#xff08;下称WUR&#xff09;联办的“第二届国际智慧温室种植挑战赛”&#xff08;下称比赛&#xff09;落幕。在全球疫情肆虐之时&#xff0c;复赛的五支队伍挑战用 …

在ASP.NET中值得注意的两个地方

在ASP.NET中ASPX页面的Page_Load事件有两个让人奇怪的地方&#xff0c;你应该记住它们&#xff1a; a.有时Page_Load事件在你的ASP.NET页面里会发生多次。这种情况发生的一个可能的原因是你把ASPX页面的AutoEvenWireup值设置成了True。如果是这样&#xff0c;那么在“Sub Page…

yii 级联删除

alter table 外键表 add constraint 级联删除名 foreign key (外键字段) references 主表名(主表字段)ON delete CASCADE go 转载于:https://blog.51cto.com/dasangshu/624605