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