C#调用WIN32API系列一列举可用COM端口
2002-12-03 08:52
471 查看
C#通过调用WIN32API可以实现非常强大的功能,本文将着重讲述如何通过调用WIN32API实现列举
所有COM端口。
首先我们看看EnumPorts函数的定义
BOOL EnumPorts(
LPTSTR pName, // server name
DWORD Level, // information level
LPBYTE pPorts, // port information buffer
DWORD cbBuf, // size of port information buffer
LPDWORD pcbNeeded, // bytes received or required
LPDWORD pcReturned // number of ports received
);
这个api返回信息放在pPorts所指的缓冲区中,是一个
struct PORT_INFO_1
{
string pName;
}
的结构数组。根据Level参数不同,这个结构数组有些不同。
C#要调用API首先要引入动态库,EnumPorts在winspool.drv这个动态库中。引入语句如下
[DllImport("winspool.drv", CharSet=CharSet.Auto)]
然后是定义PORT_INFO_1结构
struct PORT_INFO_1
{
[MarshalAs(UnmanagedType.LPTStr)]
public string pName;
}
好了,全部的源代码如下:
using System;
using System.Collections;
using System.Runtime.InteropServices;
using System.Diagnostics;
namespace rooksoft.Demo {
public class EnumPortDemo {
[StructLayout(LayoutKind.Sequential, CharSet=CharSet.Auto)]
struct PORT_INFO_1
{
[MarshalAs(UnmanagedType.LPTStr)]
public string pName;
}
[DllImport("winspool.drv", CharSet=CharSet.Auto)]
static extern bool EnumPorts(string pName, int level, IntPtr bufptr,
int cbBuf, out int pcbNeeded, out int pcReturned);
public EnumPortDemo()
{
}
public void ListPorts()//out ArrayList portslist
{
int pcReturned = 0;
int pcbNeeded = 0;
IntPtr outb = IntPtr.Zero;
EnumPorts(null, 1, outb, 0, out pcbNeeded, out pcReturned);
outb = Marshal.AllocHGlobal(pcbNeeded+1);
EnumPorts(null, 1, outb, pcbNeeded, out pcbNeeded, out pcReturned);
PORT_INFO_1[] portsArray = new PORT_INFO_1[pcReturned];
IntPtr current = outb;
for (int i=0; i<pcReturned; i++)
{
portsArray[i] = (PORT_INFO_1)
Marshal.PtrToStructure(current,
typeof(PORT_INFO_1));
current=(IntPtr)
((int)current+Marshal.SizeOf(typeof(PORT_INFO_1)));
Console.WriteLine(portsArray[i].pName);
}
Marshal.FreeHGlobal(outb);
//return manyArr;
}
public static void Main() {
EnumPortDemo demo = new EnumPortDemo();
demo.ListPorts();
}
}
}
联系作者zlyperson@163.net
所有COM端口。
首先我们看看EnumPorts函数的定义
BOOL EnumPorts(
LPTSTR pName, // server name
DWORD Level, // information level
LPBYTE pPorts, // port information buffer
DWORD cbBuf, // size of port information buffer
LPDWORD pcbNeeded, // bytes received or required
LPDWORD pcReturned // number of ports received
);
这个api返回信息放在pPorts所指的缓冲区中,是一个
struct PORT_INFO_1
{
string pName;
}
的结构数组。根据Level参数不同,这个结构数组有些不同。
C#要调用API首先要引入动态库,EnumPorts在winspool.drv这个动态库中。引入语句如下
[DllImport("winspool.drv", CharSet=CharSet.Auto)]
然后是定义PORT_INFO_1结构
struct PORT_INFO_1
{
[MarshalAs(UnmanagedType.LPTStr)]
public string pName;
}
好了,全部的源代码如下:
using System;
using System.Collections;
using System.Runtime.InteropServices;
using System.Diagnostics;
namespace rooksoft.Demo {
public class EnumPortDemo {
[StructLayout(LayoutKind.Sequential, CharSet=CharSet.Auto)]
struct PORT_INFO_1
{
[MarshalAs(UnmanagedType.LPTStr)]
public string pName;
}
[DllImport("winspool.drv", CharSet=CharSet.Auto)]
static extern bool EnumPorts(string pName, int level, IntPtr bufptr,
int cbBuf, out int pcbNeeded, out int pcReturned);
public EnumPortDemo()
{
}
public void ListPorts()//out ArrayList portslist
{
int pcReturned = 0;
int pcbNeeded = 0;
IntPtr outb = IntPtr.Zero;
EnumPorts(null, 1, outb, 0, out pcbNeeded, out pcReturned);
outb = Marshal.AllocHGlobal(pcbNeeded+1);
EnumPorts(null, 1, outb, pcbNeeded, out pcbNeeded, out pcReturned);
PORT_INFO_1[] portsArray = new PORT_INFO_1[pcReturned];
IntPtr current = outb;
for (int i=0; i<pcReturned; i++)
{
portsArray[i] = (PORT_INFO_1)
Marshal.PtrToStructure(current,
typeof(PORT_INFO_1));
current=(IntPtr)
((int)current+Marshal.SizeOf(typeof(PORT_INFO_1)));
Console.WriteLine(portsArray[i].pName);
}
Marshal.FreeHGlobal(outb);
//return manyArr;
}
public static void Main() {
EnumPortDemo demo = new EnumPortDemo();
demo.ListPorts();
}
}
}
联系作者zlyperson@163.net
相关文章推荐
- C#调用WIN32API系列一列举可用COM端口
- C#调用WIN32API系列一列举可用COM端口
- C#调用WIN32API系列二列举局网内共享打印机
- C#调用WIN32API系列二列举局网内共享打印机
- C#可以直接调用的Win32API
- C# 互操作性入门系列(三):平台调用中的数据封送处理
- vb6.0中声明Windows的Win32API时用到的结构体声明(vb.net c#中也可用)
- 关于c#调用win32api
- C#可以直接调用的Win32API
- vb6.0中声明Windows的Win32API时用到的函数声明(vb.net c#中也可用)
- C# 互操作性入门系列(二):使用平台调用调用Win32 函数
- C#互操作性入门系列(二):使用平台调用调用Win32 函数 推荐
- 在C#中能否直接调用WIN32API
- InstallShield系列(二) 调用C#生成的dll
- C# 互操作性入门系列(三):平台调用中的数据封送处理
- C#可以直接调用的Win32API(和VCL做的整理工作非常类似)
- C#编译器优化那点事 c# 如果一个对象的值为null,那么它调用扩展方法时为甚么不报错 webAPI 控制器(Controller)太多怎么办? .NET MVC项目设置包含Areas中的页面为默认启动页 (五)Net Core使用静态文件 学习ASP.NET Core Razor 编程系列八——并发处理
- 使用C#调用金诚信71x系列读卡器的DLL
- 使用C#调用金诚信71x系列读卡器的DLL
- Interop with PROPVARIANTs in .NET -- C# 调用C++ COM 系列笔记