IXMLDOMDOCUMENT 的appendchild,removechild返回E_FAIL,是的,这时候发现只能读DOM的内容,只要写入便会失败,原来WINDOWS在两个相关的处理器(如"Msxml2.XSLTemplate",以及则直接读取修改方式)内容同时处理时,XSLT处理器加载XML DOM后,应该此时对XML DOM进行了写保护(这和线程安全问题有些类似,但又不是线程安全问题)
例子代码:
HRESULT LoaderXMLDocument(IXMLDOMDocument* pXMLDocument, IMoniker* pUrlMoniker, IMoniker* pxsltMoniker, IMoniker** pNewMoniker)
{
CComPtr<IXMLDOMDocument> pMSXSLTDoc;
HRESULT hr = pMSXSLTDoc.CoCreateInstance(L"Msxml2.FreeThreadedDOMDocument");
if(pMSXSLTDoc == NULL)
return hr;
CComPtr<IBindCtx> srpBindCtx;
hr = ::CreateBindCtx(NULL, &srpBindCtx);
pMSXSLTDoc->put_async(VARIANT_FALSE);
pMSXSLTDoc->put_resolveExternals (VARIANT_FALSE);
CComQIPtr<IPersistMoniker> pMk = pMSXSLTDoc;
hr = pMk->Load(FALSE, pxsltMoniker, srpBindCtx, STGM_READ);
if(FAILED(hr))
{
return hr;
}
CComPtr<IXSLTemplate> pXSLTemplate;
hr = pXSLTemplate.CoCreateInstance(L"Msxml2.XSLTemplate");
if(FAILED(hr))
{
return hr;
}
hr = pXSLTemplate->putref_stylesheet(pMSXSLTDoc);
if(FAILED(hr))
{
return hr;
}
CComPtr<IXSLProcessor> pXSLProcessor;
pXSLTemplate->createProcessor(&pXSLProcessor);
pXSLProcessor->put_input(CComVariant(pXMLDocument));
CComPtr<IDispatch> pDispDoc;
GetHostDocumentDisp(&pDispDoc);
if(pDispDoc)
{
pXSLProcessor->addObject(pDispDoc, CComBSTR(L"urn:editordocument"));
}
CComPtr<IDispatch> pDispView;
GetHostViewDisp(&pDispView);
if(pDispView)
{
pXSLProcessor->addObject(pDispView, CComBSTR(L"urn:editorview"));
}
VARIANT_BOOL bVar = VARIANT_FALSE;
hr = pXSLProcessor->transform(&bVar);
if(FAILED(hr))
{
//this is an error
}
那么此时,pXSLProcessor相关的xslt相关调用的函数中,就不能在修改稿XML DOM了(比如pXMLDocument进行appendchild和removechild一类的写操作)