在C#中根据HardwareID获取驱动程序信息的实现代码

近日在工作中需要根据设备的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。

(0)

相关推荐

  • 在C#中根据HardwareID获取驱动程序信息的实现代码

    近日在工作中需要根据设备的HardwareID来获取设备的驱动程序信息,比如驱动程序版本等.经过摸索,得到了两种不同的解决办法,两种办法各有千秋,写出来给大家分享. 1 使用WMI中的Win32_PnPSignedDriver类 Win32_PnPSignedDriver的详细信息:http://msdn2.microsoft.com/en-us/library/aa394354.aspx 使用WMI(Windows Management Instrumentation)是最为方便的方法.可以根

  • c# 获取计算机硬件信息的示例代码

    /// <summary> /// 获取CPU的编号 /// </summary> /// <returns>CPU的编号</returns> public static List<string> GetCPUID() { List<string> lstInfo =new List<string>(); ManagementClass cimobject = new ManagementClass("Win32

  • Spring 中优雅的获取泛型信息的方法

    简介 Spring 源码是个大宝库,我们能遇到的大部分工具在源码里都能找到,所以笔者开源的 mica 完全基于 Spring 进行基础增强,不重复造轮子.今天我要分享的是在 Spring 中优雅的获取泛型. 获取泛型 自己解析 我们之前的处理方式,代码来源 vjtools(江南白衣). /** * 通过反射, 获得Class定义中声明的父类的泛型参数的类型. * * 注意泛型必须定义在父类处. 这是唯一可以通过反射从泛型获得Class实例的地方. * * 如无法找到, 返回Object.clas

  • jQuery中使用Ajax获取JSON格式数据示例代码

    JSON(JavaScript Object Notation)是一种轻量级的数据交换格式.JSONM文件中包含了关于"名称"和"值"的信息.有时候我们需要读取JSON格式的数据文件,在jQuery中可以使用Ajax或者 $.getJSON()方法实现. 下面就使用jQuery读取music.txt文件中的JSON数据格式信息. 首先,music.txt中的内容如下: 复制代码 代码如下: [ {"optionKey":"1"

  • php获取服务器信息的实现代码

    复制代码 代码如下: <html><head><meta http-equiv="content-type" content="text/html; charset=gb2312"><title>获取服务器信息</title></head><body><?php$sysos = $_SERVER["SERVER_SOFTWARE"];      //获取服务

  • spring security获取用户信息的实现代码

    前言 我们在使用spring security的时候可以通过好几种方法获取用户信息, 但是今天这篇文章介绍的是一个笔者觉得最优雅的实现; 借鉴现有的spring security controller自动注入参数的方法, 我们来进一步的实现更适合我们业务的用户信息获取方法; 思路 现在spring security会在controller自动注入Authentication/Userdetails等参数, 我们拿到这些对象之后还需要一些处理才可以拿到我们需要的信息, 例如用户ID; 那获取用户I

  • php采集文章中的图片获取替换到本地(实现代码)

    复制代码 代码如下: /** * 获取替换文章中的图片路径 * @param string $xstr 内容 * @param string $keyword 创建照片的文件名 * @param string $oriweb 网址 * @return string *  */function replaceimg($xstr,$keyword, $oriweb){ //保存路径    $d = date('Ymd', time());    $dirslsitss = '/var/www/web

  • 通过拖曳获取文件信息的bat代码分享

    前言 大家应该都有所体会,有时候我们需要使用命令行处理文件,需要定期.多次重复时,手动输入或者复制粘贴路径和文件名效率非常低下. 此时可以使用bat命令减轻工作量. bat是Windows下的批处理文件,每一行都是一条DOS命令. 获取文件属性 @echo off echo 无后缀名: %~n1 echo 有后缀名: %~nx1 echo 绝对路径: %1 echo 短路径名的绝对路径: %~s1 echo 驱动器和路径: %~dp1 echo 驱动器: %~d1 echo 路径: %~p1 e

  • Hardware_Info.vbs 获取硬件信息的VBS代码

    复制代码 代码如下: 'Hardware_Info.vbs v1.1 BY: fastslzOn Error Resume NextDim WMI,WS,FsoSet WMI = GetObject("Winmgmts:{impersonationLevel=impersonate}!\\.\root\cimv2")Set cOSs = WMI.ExecQuery("Select * from Win32_OperatingSystem")For Each oOS

  • AJAX异步从优酷专辑中采集所有视频及信息(JavaScript代码)

    个人觉得不甚满意,因为VB.NET在.NET环境下执行采集,产生的网络连接与正则匹配消耗太大,而我最终的应用是在视频采集提交上,所以就考虑用JS的AJAX+正则表达式来完成这个目标. 以前一直没有系统地学习JS和正则,所以花费的时间比预想的要长,折腾了一个晚上,糊弄出了下面的代码.=..= 复制代码 代码如下: <!-- 夜闻香原创,转载请保留此信息,万分感谢! 博客: http://clso.cnblogs.com 主页: http://cleclso.cn QQ:315514678 E-ma

随机推荐