3

ChildNodes.FindNode('') ( 1 ) が失敗するのに ChildNodes[''] ( 2 ) が成功する理由を誰か説明できますか? ==>

コード:

const
   cNodeSOAPEnvelope  = 's:Envelope';
   cNodeSOAPBody      = 's:Body';
   cNodeSOAPBodyFault = 's:Fault';
   cNodeSOAPHeader    = 's:Header';
   cNodeFaultCode     = 'faultcode';
   cNodeFaultString   = 'faultstring';

procedure TFrmXMLParsingTests.BtnTestClick(Sender: TObject);
var
   SoapBodyNode,
   SoapBodyFaultNode,
   SoapHeaderNode,
   FaultCodeNode,
   FaultTextNode,
   RootNode: IXmlNode;
begin
   RootNode := XMLDoc.DocumentElement;
   Assert(RootNode.NodeName = cNodeSOAPEnvelope,'Root node is not SOAP envelope');
   SoapBodyNode := RootNode.ChildNodes[cNodeSOAPBody];
   SoapBodyFaultNode := SoapBodyNode.ChildNodes[cNodeSOAPBodyFault];
   if SoapBodyFaultNode <> nil then
   begin
//      FaultCodeNode := SoapBodyFaultNode.ChildNodes.FindNode(cNodeFaultCode);     (*1*)
//      FaultTextNode := SoapBodyFaultNode.ChildNodes.FindNode(cNodeFaultString);   (*1*)
      FaultCodeNode := SoapBodyFaultNode.ChildNodes[cNodeFaultCode];                (*2*) 
      FaultTextNode := SoapBodyFaultNode.ChildNodes[cNodeFaultString];              (*2*) 
      Exit; // Nothing more to do
   end; { fault in body }
   SoapHeaderNode := RootNode.ChildNodes.FindNode(cNodeSOAPHeader);
   Assert(SoapHeaderNode <> nil,'SOAP Header not found');
end;

XML:

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
   <s:Body>
      <s:Fault>
         <faultcode xmlns:a="http://schemas.microsoft.com/exchange/services/2006/types">a:ErrorSchemaValidation</faultcode>
         <faultstring xml:lang="nl-NL">The [snip] value.</faultstring>
         <detail>
            <e:ResponseCode xmlns:e="http://schemas.microsoft.com/exchange/services/2006/errors">ErrorSchemaValidation</e:ResponseCode>
            <e:Message xmlns:e="http://schemas.microsoft.com/exchange/services/2006/errors">The request failed schema validation.</e:Message>
            <t:MessageXml xmlns:t="http://schemas.microsoft.com/exchange/services/2006/types">
               <t:LineNumber>10</t:LineNumber>
               <t:LinePosition>30</t:LinePosition>
               <t:Violation>The [snip] value.</t:Violation>
            </t:MessageXml>
         </detail>
      </s:Fault>
   </s:Body>
</s:Envelope>

ありがとう
ヤン

4

2 に答える 2

3

ドキュメントを読んでください。存在しないノードを要求するプロパティにアクセスすると、親 XMLDocument のプロパティでフラグが有効になってChildNodes[]いる場合は、新しいノードが自動的に作成されます。そうでない場合は、代わりに例外が発生します。doNodeAutoCreateOptionsEXMLDocError

更新ChildNodes[]: ドキュメントには複数の XML 名前空間が含まれているため、プロパティとFindNode()メソッドを使用するときは名前空間を考慮する必要があります。コードが<faultcode>and<faultstring>ノードを正しく見つけられない理由は、それらの名前空間がタグとは異なる (実際には名前空間がまったくない) ためです<Fault>が、ChildNodes[]プロパティと 1 パラメーターFindNode()メソッドでは、それらが親ノードと同じ名前空間。別の名前空間を持つ子ノードの名前空間ノードを検索する場合、子ノードを正しく見つけるために、子の名前空間を検索に含める必要があります。

これを試して:

uses
  ..., XmlIntf, XMLDom, XmlDoc;

const
  cSoapEnvelopeNS    = 'http://schemas.xmlsoap.org/soap/envelope/';
  cExchangeErrorsNS  = 'http://schemas.microsoft.com/exchange/services/2006/errors';
  cExchangeTypesNS   = 'http://schemas.microsoft.com/exchange/services/2006/types';

  cNodeSOAPEnvelope  = 'Envelope';
  cNodeSOAPBody      = 'Body';
  cNodeSOAPBodyFault = 'Fault';
  cNodeSOAPHeader    = 'Header';
  cNodeFaultCode     = 'faultcode';
  cNodeFaultString   = 'faultstring';
  cNodeDetail        = 'detail';
  cNodeResponseCode  = 'ResponseCode';
  cNodeMessage       = 'Message';
  cNodeMessageXml    = 'MessageXml';
  cNodeLineNumber    = 'LineNumber';
  cNodeLinePosition  = 'LinePosition';
  cNodeViolation     = 'Violation';

procedure TFrmXMLParsingTests.BtnTestClick(Sender: TObject);
var
   RootNode,
   SoapBodyNode,
   SoapBodyFaultNode,
   SoapHeaderNode,
   FaultCodeNode,
   FaultTextNode,
   DetailNode,
   ResponseCodeNode,
   MessageNode,
   MessageXmlNode,
   LineNumberNode,
   LinePositionNode,
   ViolationNode: IXMLNode;
begin
   RootNode := XMLDoc.DocumentElement;
   Assert(NodeMatches(RootNode.DOMNode, cNodeSOAPEnvelope, cSoapEnvelopeNS), 'Root node is not SOAP envelope');

   // these have the same namespace as <Envelope>
   SoapBodyNode := RootNode.ChildNodes[cNodeSOAPBody];
   SoapBodyFaultNode := SoapBodyNode.ChildNodes.FindNode(cNodeSOAPBodyFault);

   if SoapBodyFaultNode <> nil then
   begin
      // these have a different namespace than <Fault>!
      FaultCodeNode := SoapBodyFaultNode.ChildNodes.FindNode(cNodeFaultCode, '');
      FaultTextNode := SoapBodyFaultNode.ChildNodes.FindNode(cNodeFaultString, '');
      DetailNode := SoapBodyFaultNode.ChildNodes.FindNode(cNodeDetail, '');

      if DetailNode <> nil then
      begin
        // these have different namespaces than <detail>!
        ResponseCodeNode := DetailNode.ChildNodes.FindNode(cNodeResponseCode, cExchangeErrorsNS);
        MessageNode := DetailNode.ChildNodes.FindNode(cNodeMessage, cExchangeErrorsNS);
        MessageXmlNode := DetailNode.ChildNodes.FindNode(cNodeMessageXml, cExchangeTypesNS);

        if MessageXmlNode <> nil then
        begin
          // these have the same namespace as <MessageXml>
          LineNumberNode := MessageXmlNode.ChildNodes.FindNode(cNodeLineNumber);
          LinePositionNode := MessageXmlNode.ChildNodes.FindNode(cNodeLinePosition);
          ViolationNode := MessageXmlNode.ChildNodes.FindNode(cNodeViolation);
        end;
      end;
      Exit;
   end;

   // this has the same namespace as <Envelope>
   SoapHeaderNode := RootNode.ChildNodes.FindNode(cNodeSOAPHeader);
   Assert(SoapHeaderNode <> nil,'SOAP Header not found');
end;
于 2012-11-14T18:39:21.817 に答える
0

独自の関数を作成します。

function SearchChilds(StartNode: IXMLNode; Nodename: string): IXMLnode;
var i: Integer;
begin
   Result := nil;
   for i := 0 to StartNode.ChildNodes.Count-1 do
   begin
      if (StartNode.ChildNodes[i] <> nil) and (StartNode.ChildNodes[i].LocalName = Nodename) then
      begin
         Result := StartNode.ChildNodes[i];
         break;
      end;
      if StartNode.ChildNodes[i].HasChildNodes then
      begin
         Result := SearchChilds(StartNode.ChildNodes[i], Nodename);
         if Result<>nil then break;
      end;
   end;
end;
于 2013-02-27T16:55:54.930 に答える