在C#中根據(jù)HardwareID獲取驅(qū)動程序信息的實現(xiàn)代碼
近日在工作中需要根據(jù)設(shè)備的HardwareID來獲取設(shè)備的驅(qū)動程序信息,比如驅(qū)動程序版本等。經(jīng)過摸索,得到了兩種不同的解決辦法,兩種辦法各有千秋,寫出來給大家分享。
1 使用WMI中的Win32_PnPSignedDriver類
Win32_PnPSignedDriver的詳細信息:http://msdn2.microsoft.com/en-us/library/aa394354.aspx
使用WMI(Windows Management Instrumentation)是最為方便的方法??梢愿鶕?jù)下面的程序片段來得到我們所需要的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";
}
這樣取得驅(qū)動程序的方式是非常簡潔的,但是有一個非常嚴重的問題就是效率問題。平均說來,每執(zhí)行一次查詢,得到一個DriverVersion需要大約3秒的時間。對于我們的應(yīng)用來說,這個時間是不可以接受的。也許你會說,為什么不用更多的限定符號來進一步減少查詢的次數(shù)呢?
如果我們把連接字符串改成:
string queryString = "SELECT HardwareID, DriverVersion FROM Win32_PnPSignedDriver WHERE HardwareID='somehardware'";
程序的效率并沒有明顯的改進。而且還發(fā)現(xiàn)一個問題,如果我們somehardware里面含有一個'\'(也就是HardwareID='some\\hardware'),那么一定會得到一個“Invalid Query”異常。但是在WMITOOLS里面查詢又是正常的,希望達人出來指點一下。最后根據(jù)MSDN的描述,只有Windows Vista,Windows XP和Windows 2003支持這個類。由于我們的程序需要跑在2000下,因此這種方法是行不通的了。
2 使用PInvoke
由于無法使用WMI,因此就想到了使用PInvoke的方式調(diào)用Windows API。通過查詢MSDN,知道可以使用SetupDixxxx這種函數(shù)來實現(xiàn)我們的功能?;镜乃悸啡缦拢?br />
(1)利用SetupDiGetClassDevs這個函數(shù)得到一個含有所有設(shè)備信息的類。
(2)利用SetupDiEnumDeviceInfo得到某個具體設(shè)備的信息,保存在一個名為SP_DEVINFO_DATA的結(jié)構(gòu)中。
(3)利用SetupDiGetDeviceRegistryProperty得到設(shè)備的HardwareID,和輸入的HardwareID比較
(4)如果兩個HardwareID是一樣的,那么就利用SetupDiBuildDriverInfoList得到這個設(shè)備的驅(qū)動程序信息列表
(5)利用SetupDiEnumDriverInfo遍歷驅(qū)動程序信息列表,得到所有需要的信息,保存在一個名為SP_DRVINFO_DATA的結(jié)構(gòu)中
(6)從SP_DRVINFO_DATA中就可以得到驅(qū)動程序的版本。是一個DWORDLONG類型的數(shù),需要轉(zhuǎn)換成x.x.x.x的結(jié)構(gòu)
要值得注意的是上述函數(shù)都封裝在setupapi.dll中,要使用這些函數(shù),需要安裝Windows DDK。
在C#中,我們利用pInvoke的方式來調(diào)用Windows API的時候,需要注意類型的對應(yīng)和結(jié)構(gòu)對齊。比如上面的SP_DEVINFO_DATA結(jié)構(gòu)需要按照如下方式聲明
[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。如果不按照這樣聲明,無法調(diào)用成功。
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轉(zhuǎn)換成x.x.x.x的版本,可以按照下面的方式轉(zhuǎn)換。DWORDLONG是8字節(jié)的無符號整數(shù),x.x.x.x中的x是從0到65536的無符號整數(shù),占2個字節(jié)。因此可以直接把8字節(jié)的整數(shù)分成4個2字節(jié)的整數(shù),最后合起來就是版本號了。假設(shè)版本version = 1407379348914176,將version轉(zhuǎn)換成2進制數(shù)為:
101 00000000 00000001 00001010 00101000 00000000 00000000
--- --------------------- ---------------------- ---------------------
5 1 2600 0
因此,可以得到版本是5.1.2600.0。
可以用下面這個示例函數(shù)來得到版本信息
//version = 1407379348914176,轉(zhuǎn)換后的版本為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();
}
通過調(diào)用API這種方式,速度得到了很大的提高,1秒之內(nèi)就可以完成一次查詢。而且適合于Win2000,Win XP,Win2003和Vista。
相關(guān)文章
C#多線程之Thread中Thread.Join()函數(shù)用法分析
這篇文章主要介紹了C#多線程之Thread中Thread.Join()函數(shù)用法,實例分析了Thread.Join()方法的原理與使用技巧,非常具有實用價值,需要的朋友可以參考下2015-04-04
將文件夾下所有文件輸出到日志文件中 c#遞歸算法學(xué)習(xí)示例
這篇文章主要介紹了將文件夾下所有文件輸出到日志文件中,通過這個示例我們學(xué)習(xí)一下遞歸算法的使用方法2014-01-01

