读书人

关于C++ 访问WMI 程序11,该如何解决

发布时间: 2012-08-14 10:39:57 作者: rapoo

关于C++ 访问WMI 程序11
我自己写了一个C++程序 (vs2010),老是连接不上Server呢 怎也找不到原因。

下面的if语句里 执行完 不等于S_OK 导致下面的语句执行不了
string wsNamespace = ("root\\cimv2");
if(pWbemLocator->ConnectServer((BSTR)&wsNamespace,NULL,NULL,NULL,0,NULL,NULL,&pWbemServices) ==S_OK)

然后我在网上 找了一段代码复制下来,还是执行不了 也是在ConnectServer 这里卡住了,我修改的唯一的地方就是

定义了string 类型的namespace 然后 在函数参数里强制转换成BSTR类型 跟上面的一样

网上复制的代码 如下:
//

#include "stdafx.h"
#include<iostream>
#include <Wbemidl.h>
# pragma comment(lib, "wbemuuid.lib")

using namespace std;

//第一步:初始化COM前面已经讲过,WMI是基于COM(组件对象模型)的,所以在使用WMI前,我们必须首先初始化COM。这里主要用到两个函数
int GetInfo()
{
HRESULT hres;

//初始化 COM.

hres = CoInitializeEx(0, COINIT_MULTITHREADED);

if (FAILED(hres))

{

cout << "Failed to initialize COM library. "

<< "Error code = 0x"

<< hex << hres << endl;

return 1; // Program has failed.

}

// 设置进程安全级别

hres = CoInitializeSecurity(

NULL,

-1, // COM negotiates service

NULL, // Authentication services

NULL, // Reserved

RPC_C_AUTHN_LEVEL_DEFAULT, // authentication

RPC_C_IMP_LEVEL_IMPERSONATE, // Impersonation

NULL, // Authentication info

EOAC_NONE, // Additional capabilities

NULL // Reserved

);

if (FAILED(hres))

{
cout << "Failed to initialize security. "
<< "Error code = 0x"
<< hex << hres << endl;
CoUninitialize();
return 1; // Program has failed.
}

//
//第二步:创建一个WMI命名空间连接。
//
//WMI最大特点就是使用了统一的命名空间

//创建一个CLSID_WbemLocator对象

IWbemLocator *pLoc = 0;
hres = CoCreateInstance(
CLSID_WbemLocator,
0,
CLSCTX_INPROC_SERVER,
IID_IWbemLocator, (LPVOID *) &pLoc);

if (FAILED(hres))
{
cout << "Failed to create IWbemLocator object. "
<< "Error code = 0x"
<< hex << hres << endl;

CoUninitialize();

return 1; // Program has failed.
}

IWbemServices *pSvc = 0;

//使用pLoc连接到” root\cimv2” 并把pSvc的指针也搞定了

string NameSpace = "ROOT\\CIMV2";//这里我自己定义的string类型 在下面强制转换成BSTR类型

hres = pLoc->ConnectServer(
(BSTR)&NameSpace, // WMI namespace
NULL, // User name
NULL, // User password
0, // Locale
NULL, // Security flags
0, // Authority
0, // Context object
&pSvc // IWbemServices proxy
);

if (FAILED(hres))
{
cout << "Could not connect. Error code = 0x"


<< hex << hres << endl;

pLoc->Release();

CoUninitialize();

return 1; // Program has failed.

}

//已经连接到WMI了

cout << "Connected to ROOT\\CIMV2 WMI namespace" << endl;

//第三步:设置连接的安全级别。
//
// 为什么要这么做?我也不清楚,MS说要这样那就这样吧:)
//
hres = CoSetProxyBlanket(

pSvc, // the proxy to set

RPC_C_AUTHN_WINNT, // authentication service

RPC_C_AUTHZ_NONE, // authorization service

NULL, // Server principal name

RPC_C_AUTHN_LEVEL_CALL, // authentication level

RPC_C_IMP_LEVEL_IMPERSONATE, // impersonation level

NULL, // client identity

EOAC_NONE // proxy capabilities

);



if (FAILED(hres))

{

cout << "Could not set proxy blanket. Error code = 0x"

<< hex << hres << endl;

pSvc->Release();

pLoc->Release();

CoUninitialize();

return 1; // Program has failed.

}

//第四步:执行你的代码,达成你的目的。
//
//这里一不小心就要引入WQL这个概念了。
//
// WQL就是WMI中的查询语言,WQL的全称是WMI Query Language,简称为WQL,翻译成中文好像可以成为Windows管理规范查询语言。熟悉SQL语言的朋友会感觉它和SQL非常相似。
//
//
//
//WQL其实非常简单,它有如下特点:
//
//1、每个WQL语句必须以SELECT开始;
//
//2、SELECT后跟你需要查询的属性名(我刚才对应SQL将其称之为字段名了),也可以像SQL一样,以*表示返回所有属性值;
//
//3、FROM关键字;
//
//4、你要查询的类的名字;

//这里是列出正在运行的进程的例子

//为了接收结果,你必须定义一个枚举对象

IEnumWbemClassObject* pEnumerator = NULL;

string Lan = "WQL";
string SelectStr = "SELECT * FROM Win32_Process";//这里也是string转换BSTR

hres = pSvc->ExecQuery(

(BSTR)&Lan,

(BSTR)&SelectStr,

WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,

NULL,

&pEnumerator);

if (FAILED(hres))
{
cout << "Query for processes failed. "
<< "Error code = 0x"
<< hex << hres << endl;

pSvc->Release();

pLoc->Release();

CoUninitialize();

return 1; // Program has failed.

}

else

{

IWbemClassObject *pclsObj;

ULONG uReturn = 0;



while (pEnumerator)

{

// 推出下一个对象

hres = pEnumerator->Next(WBEM_INFINITE, 1,

&pclsObj, &uReturn);

//没有东西了就跳出去吧

if(0 == uReturn)

{

break;

}



VARIANT vtProp;



// Get the value of the Name property

hres = pclsObj->Get(L"Name", 0, &vtProp, 0, 0);

wcout << "Process Name : " << vtProp.bstrVal << endl;

VariantClear(&vtProp);



}



}

//第五步:清除关闭你的程序。
//
//释放掉该释放的东西是个好习惯。

pSvc->Release();

pLoc->Release();

CoUninitialize();
}


int _tmain(int argc, _TCHAR* argv[])
{
GetInfo();
return 0;
}



[解决办法]
hres = CoInitializeEx(0, COINIT_MULTITHREADED);
你确定是多线程套间的?
一般就是使用CoInitialize(NULL);

wstring NameSpace = L"ROOT\\CIMV2";//这里我自己定义的string类型 在下面强制转换成BSTR类型

hres = pLoc->ConnectServer(
(BSTR)&NameSpace, // WMI namespace
NULL, // User name
NULL, // User password
0, // Locale
NULL, // Security flags
0, // Authority
0, // Context object
&pSvc // IWbemServices proxy
);

记得BSTR是宽字节的

读书人网 >C++

热点推荐