在C#中根据HardwareID获取驱动程序信息的实现代码
作者:mdxy-dxy 发布时间:2023-08-04 08:15:31
近日在工作中需要根据设备的HardwareID来获取设备的驱动程序信息,比如驱动程序版本等。经过摸索,得到了两种不同的解决办法,两种办法各有千秋,写出来给大家分享。
1 使用WMI中的Win32_PnPSignedDriver类
Win32_PnPSignedDriver的详细信息:http://msdn2.microsoft.com/en-us/library/aa394354.aspx
使用WMI(Windows Management Instrumentation)是最为方便的方法。可以根据下面的程序片段来得到我们所需要的DriverVersion。
private string GetDriverVersion( string hardwareID )
{
string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver";
SelectQuery selectQuery = new SelectQuery( queryString );
ManagementObjectSearcher searcher = new ManagementObjectSearcher(selectQuery);
foreach (ManagementObject mo in searcher.Get())
{
object tempID = mo["HardwareID"];
if( tempID!=null && tempID.ToString().ToUpper() == hardwareID.Trim().ToUpper() )
{
return mo["DriverVersion"].ToString();
}
}
return "UnknownVersion";
}
这样取得驱动程序的方式是非常简洁的,但是有一个非常严重的问题就是效率问题。平均说来,每执行一次查询,得到一个DriverVersion需要大约3秒的时间。对于我们的应用来说,这个时间是不可以接受的。也许你会说,为什么不用更多的限定符号来进一步减少查询的次数呢?
如果我们把连接字符串改成:
string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver WHERE HardwareID='somehardware'";
程序的效率并没有明显的改进。而且还发现一个问题,如果我们somehardware里面含有一个'\'(也就是HardwareID='some\\hardware'),那么一定会得到一个“Invalid Query”异常。但是在WMITOOLS里面查询又是正常的,希望达人出来指点一下。最后根据MSDN的描述,只有Windows Vista,Windows XP和Windows 2003支持这个类。由于我们的程序需要跑在2000下,因此这种方法是行不通的了。
2 使用PInvoke
由于无法使用WMI,因此就想到了使用PInvoke的方式调用Windows API。通过查询MSDN,知道可以使用SetupDixxxx这种函数来实现我们的功能。基本的思路如下:
(1)利用SetupDiGetClassDevs这个函数得到一个含有所有设备信息的类。
(2)利用SetupDiEnumDeviceInfo得到某个具体设备的信息,保存在一个名为SP_DEVINFO_DATA的结构中。
(3)利用SetupDiGetDeviceRegistryProperty得到设备的HardwareID,和输入的HardwareID比较
(4)如果两个HardwareID是一样的,那么就利用SetupDiBuildDriverInfoList得到这个设备的驱动程序信息列表
(5)利用SetupDiEnumDriverInfo遍历驱动程序信息列表,得到所有需要的信息,保存在一个名为SP_DRVINFO_DATA的结构中
(6)从SP_DRVINFO_DATA中就可以得到驱动程序的版本。是一个DWORDLONG类型的数,需要转换成x.x.x.x的结构
要值得注意的是上述函数都封装在setupapi.dll中,要使用这些函数,需要安装Windows DDK。
在C#中,我们利用pInvoke的方式来调用Windows API的时候,需要注意类型的对应和结构对齐。比如上面的SP_DEVINFO_DATA结构需要按照如下方式声明
[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)]
public struct SP_DEVINFO_DATA
{
public int cbSize;
public Guid ClassGuid;
public IntPtr DevInst;
public IntPtr Reserved;
}
要注意的是LayoutKind.Sequential, Pack = 4 和 public IntPtr Reserved。如果不按照这样声明,无法调用成功。
SP_DRVINFO_DATA也可以按照一样的方式进行声明。
[StructLayout(LayoutKind.Sequential, Pack = 4, CharSet = CharSet.Auto)]
public struct SP_DRVINFO_DATA
{
public int cbSize;
public int DriverType;
public IntPtr Reserved;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
public string Description;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
public string MfgName;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
public string ProviderName;
public FILETIME DriverDate;
public ulong DriverVersion;
}
对于最后的从DWORDLONG转换成x.x.x.x的版本,可以按照下面的方式转换。DWORDLONG是8字节的无符号整数,x.x.x.x中的x是从0到65536的无符号整数,占2个字节。因此可以直接把8字节的整数分成4个2字节的整数,最后合起来就是版本号了。假设版本version = 1407379348914176,将version转换成2进制数为:
101 00000000 00000001 00001010 00101000 00000000 00000000
--- --------------------- ---------------------- ---------------------
5 1 2600 0
因此,可以得到版本是5.1.2600.0。
可以用下面这个示例函数来得到版本信息
//version = 1407379348914176,转换后的版本为5.1.2600.0
private string GetVersionFromLong( ulong version )
{
ulong baseNumber = 0xFFFF;
StringBuilder sb = new StringBuilder();
ulong temp = 0L;
for( int offset = 48; offset >= 0; offset -= 16 )
{
temp = (version >> offset) & baseNumber;
sb.Append( temp.ToString() + "." );
}
return sb.ToString();
}
通过调用API这种方式,速度得到了很大的提高,1秒之内就可以完成一次查询。而且适合于Win2000,Win XP,Win2003和Vista。


猜你喜欢
- 泛型泛型的语法定义class 类名 <泛型标识,泛型标识,…>{ private 泛型标识1,变量名;常用
- DataSet 对象是支持 ADO.NET的断开式、分布式数据方案的核心对象 ,用途非常广泛.我们很多时候需要使用其中的数据,比如取得一个D
- 在Android实现没有标题栏的方法有两种:在代码中添加requestWindowFeature(Window.FEATURE_NO_TIT
- 最近在学习AngularJS的知识,收获不少,不过因为自己平时工作时开发都是用的freemarker+springmvc来做的页面数据交互,
- 1.剖析异或运算(^) 二元 ^ 运算符是为整型和 bool 类型预定义的。对于整型,^ 将计算操作数的按位“异或”。对于 bool 操作数
- 前言我们在很多博客中都有发现,Seata AT模式里面的全局锁其实是行锁,这也是Seata AT模式和XA模式在锁粒度上的最大区别。我们可以
- 概述Spring boot 中的 @Conditional 注解是一个不太常用到的注解,但确实非常的有用,我们知道 Spring Boot
- 首先介绍下JSON的定义,JSON是JavaScript Object Notation的缩写。一种轻量级的数据交换格式,具有良好的可读和便
- 本文介绍在使用C#开发WinForm窗体程序时,如何设置窗体的大小不能被改变。我们在开发一个窗体(WinForm)程序时,所有的控件都部署在
- 一、Javassist入门(一)Javassist是什么Javassist是可以动态编辑Java字节码的类库。它可以在Java程序运行时定义
- 1、什么是FeignFeign 是 Spring Cloud Netflix 组件中的一个轻量级 RESTful 的 HTTP 服务客户端,
- 起因最近在写CRUD的时候,发现有个分页的VO写的健壮性比较差,一时手痒改了一下,没想到改了之后好几个功能都出现了问题。原VO关键代码如下:
- JRebel 介绍IDEA上原生是不支持热部署的,一般更新了 Java 文件后要手动重启 Tomcat 服务器,才能生效,浪费不少生命啊。目
- public class CrossSum{ public static void main(String args[]){
- 单例模式为什么要用单例确保某个类只有一个对象,常用于访问数据库操作,服务的配置文件等。单例的关键点1、默认构造函数为private,复制构造
- 文件的上传与下载(一)在实现文件上传和下载之前我们需要做一些准备工作,在Apache官网去下载文件上传下载的两个组件,下载链接这里给出:co
- 1.1、Web Service基本概念Web Service也叫XML Web Service WebService是一种可以接收从Inte
- 本文实例为大家分享了Android仿京东左侧分类条目效果的具体代码,供大家参考,具体内容如下import android.app.Activ
- 通常在C#的实际开发过程中,会发现设置其属性ScriptErrorsSuppressed无法达到屏蔽脚本错误效果,但是可以通过下面两种方式实
- java 进制转换实例详解十进制转成十六进制: