如何使用C应用程序从WMI获取数据?

我有一个纯C应用程序发出IOCTL调用我的适配器驱动程序并显示信息,然而这是使用Visual Developer Studio 5(非托管代码)编译…我需要从我的适配器使用WMI获取一些信息.. ..我的谷歌搜索努力表明,我需要使用COM编写一个C ++应用程序来实现与wMI或带有.NET应用程序的C#的任何forms的通信a)情况确实如此吗? 没有解决我的C应用程序? b)如果以上是真的,那么我需要进行项目/ wp /工作区设置的最小级别更改是什么?

谢谢索姆

您可以从C调用COM。语法不如C ++友好,但它可以工作。 COM最初设计为使用C或C ++,COM和WMI头文件中包含本机C语言支持。 但是很长一段时间……你的程序将负责分配所有必要的对象,检查每个COM调用的错误条件,以及释放它实例化的对象。

在使用用C ++编写的文档时,转换表单的COM调用:

pSomething->Method(arg1, ...); // C++ 

至:

 pSomething->lpVtbl->Method(pSomething, arg1, ...); // C 

下面是我可以从WMI中实际提取一些信息的最短C代码。 如果成功,它应列出计算机上的处理器以及以MHz为单位的时钟频率。 该程序负责处理它分配的资源,但它不会进行任何错误检查(您应该在继续每个步骤之前查看这些hr值)。

这是一个visual studio 2008“Win32控制台应用程序”,主文件重命名为.c扩展名,并删除了额外的stdafx文件。 要使程序链接,请确保在项目属性中的Configuration Properties / Linker / Input / Additional Dependencies下包含wbemuuid.lib。 它在我的Vista盒子上成功运行。

 #define _WIN32_WINNT 0x0400 #define _WIN32_DCOM #include  #include  #include  #include  void _tmain(int argc, _TCHAR* argv[]) { // result code from COM calls HRESULT hr = 0; // COM interface pointers IWbemLocator *locator = NULL; IWbemServices *services = NULL; IEnumWbemClassObject *results = NULL; // BSTR strings we'll use (http://msdn.microsoft.com/en-us/library/ms221069.aspx) BSTR resource = SysAllocString(L"ROOT\\CIMV2"); BSTR language = SysAllocString(L"WQL"); BSTR query = SysAllocString(L"SELECT * FROM Win32_Processor"); // initialize COM hr = CoInitializeEx(0, COINIT_MULTITHREADED); hr = CoInitializeSecurity(NULL, -1, NULL, NULL, RPC_C_AUTHN_LEVEL_DEFAULT, RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE, NULL); // connect to WMI hr = CoCreateInstance(&CLSID_WbemLocator, 0, CLSCTX_INPROC_SERVER, &IID_IWbemLocator, (LPVOID *) &locator); hr = locator->lpVtbl->ConnectServer(locator, resource, NULL, NULL, NULL, 0, NULL, NULL, &services); // issue a WMI query hr = services->lpVtbl->ExecQuery(services, language, query, WBEM_FLAG_BIDIRECTIONAL, NULL, &results); // list the query results if (results != NULL) { IWbemClassObject *result = NULL; ULONG returnedCount = 0; // enumerate the retrieved objects while((hr = results->lpVtbl->Next(results, WBEM_INFINITE, 1, &result, &returnedCount)) == S_OK) { VARIANT name; VARIANT speed; // obtain the desired properties of the next result and print them out hr = result->lpVtbl->Get(result, L"Name", 0, &name, 0, 0); hr = result->lpVtbl->Get(result, L"MaxClockSpeed", 0, &speed, 0, 0); wprintf(L"%s, %dMHz\r\n", name.bstrVal, speed.intVal); // release the current result object result->lpVtbl->Release(result); } } // release WMI COM interfaces results->lpVtbl->Release(results); services->lpVtbl->Release(services); locator->lpVtbl->Release(locator); // unwind everything else we've allocated CoUninitialize(); SysFreeString(query); SysFreeString(language); SysFreeString(resource); } 

另一个选项是,如果要保持对现有C应用程序的影响较小,则编写一个DLL,在内部可以使用C ++和COM包装类来查询所需的WMI信息。

此DLL可以提供纯C接口以适应您的应用程序。 这就是我想要的方式。