介绍
MSXML (Microsoft XML Core Services) 是微软提供的一组服务,允许应用程序基于Windows平台构建基于XML的解决方案。它提供了对XML文档的加载、解析、验证、查询和转换等功能。MSXML支持多种XML标准,包括XML 1.0、XSLT 1.0、XPath 1.0、SAX2和DOM。
在本指南中,我将详细介绍如何在C++中使用MSXML库,包括初始化COM、创建XML文档对象、加载和保存XML文件、遍历节点、修改内容以及执行XPath查询等操作。
准备工作
要使用MSXML,你需要:
- Windows操作系统
- Visual Studio开发环境
- 包含MSXML头文件和链接相关库
首先,确保你的项目设置了正确的包含路径和库依赖:
#include <windows.h>
#include <msxml6.h> // 使用MSXML 6.0版本
#pragma comment(lib, "msxml6.lib")
初始化COM
由于MSXML是基于COM的技术,使用前需要初始化COM库:
#include <objbase.h>
// 初始化COM库
HRESULT hr = CoInitialize(NULL);
if (FAILED(hr)) {
printf("Failed to initialize COM library. Error code = 0x%08x\n", hr);
return -1;
}
创建XML文档对象
IXMLDOMDocument2* pXMLDoc = NULL;
// 创建MSXML DOM文档对象
hr = CoCreateInstance(CLSID_DOMDocument60, NULL, CLSCTX_INPROC_SERVER,
IID_IXMLDOMDocument2, (void**)&pXMLDoc);
if (FAILED(hr) || pXMLDoc == NULL) {
printf("Failed to create DOM document object. Error code = 0x%08x\n", hr);
CoUninitialize();
return -1;
}
// 设置异步加载为false
hr = pXMLDoc->put_async(VARIANT_FALSE);
if (FAILED(hr)) {
printf("Failed to set async property. Error code = 0x%08x\n", hr);
pXMLDoc->Release();
CoUninitialize();
return -1;
}
// 设置验证为true
hr = pXMLDoc->put_validateOnParse(VARIANT_TRUE);
if (FAILED(hr)) {
printf("Failed to set validateOnParse property. Error code = 0x%08x\n", hr);
pXMLDoc->Release();
CoUninitialize();
return -1;
}
加载XML文件
VARIANT_BOOL vbSuccess;
VARIANT varFileName;
// 初始化VARIANT变量
VariantInit(&varFileName);
varFileName.vt = VT_BSTR;
// 假设我们要加载的XML文件名为"example.xml"
varFileName.bstrVal = SysAllocString(L"example.xml");
// 加载XML文件
hr = pXMLDoc->load(varFileName, &vbSuccess);
if (FAILED(hr) || vbSuccess != VARIANT_TRUE) {
printf("Failed to load XML file. Error code = 0x%08x\n", hr);
// 获取详细的错误信息
IXMLDOMParseError* pXMLError = NULL;
BSTR bstrError = NULL;
hr = pXMLDoc->get_parseError(&pXMLError);
if (SUCCEEDED(hr)) {
pXMLError->get_reason(&bstrError);
printf("Error reason: %S\n", bstrError);
SysFreeString(bstrError);
pXMLError->Release();
}
VariantClear(&varFileName);
pXMLDoc->Release();
CoUninitialize();
return -1;
}
// 清理VARIANT变量
VariantClear(&varFileName);
遍历XML节点
// 获取文档元素(根节点)
IXMLDOMElement* pRootElem = NULL;
hr = pXMLDoc->get_documentElement(&pRootElem);
if (FAILED(hr) || pRootElem == NULL) {
printf("Failed to get document element. Error code = 0x%08x\n", hr);
pXMLDoc->Release();
CoUninitialize();
return -1;
}
// 获取根节点名称
BSTR bstrRootName;
hr = pRootElem->get_nodeName(&bstrRootName);
if (SUCCEEDED(hr)) {
printf("Root element name: %S\n", bstrRootName);
SysFreeString(bstrRootName);
}
// 遍历子节点
IXMLDOMNodeList* pNodeList = NULL;
hr = pRootElem->get_childNodes(&pNodeList);
if (SUCCEEDED(hr)) {
long length;
pNodeList->get_length(&length);
for (long i = 0; i < length; i++) {
IXMLDOMNode* pNode = NULL;
hr = pNodeList->get_item(i, &pNode);
if (SUCCEEDED(hr) && pNode != NULL) {
DOMNodeType nodeType;
pNode->get_nodeType(&nodeType);
BSTR bstrNodeName;
pNode->get_nodeName(&bstrNodeName);
if (nodeType == NODE_ELEMENT) {
printf("Element node: %S\n", bstrNodeName);
// 获取元素属性
IXMLDOMNamedNodeMap* pAttributes = NULL;
pNode->get_attributes(&pAttributes);
if (pAttributes != NULL) {
long attrCount;
pAttributes->get_length(&attrCount);
for (long j = 0; j < attrCount; j++) {
IXMLDOMNode* pAttrNode = NULL;
pAttributes