2015-06-12 39 views
0

编辑:这与链接的发布不完全相同。我遇到的主要问题是将子节点添加到空的XML节点。直接选择节点时,它将返回一个System.String类型,该类型没有AppendChild方法。解决方法是选择所有的子节点,然后按照Dan的建议在下面按名称过滤。将XML文档合并到空的XML节点中?

$emptyNode= $root.ChildNodes | ? { $_.name -eq "customers" } 

我主要使用PowerShell,但很多代码的下面我和使用.NET系统对象的工作。我想要做的是通过一个例子来解释。说我有三个XML文件:

<!-- XML File A --> 
<customer> 
    <name>ACME Co</name> 
    <users> 
     <user> 
      <name>Alex</name> 
      <age>20</age> 
     </user> 
     <user> 
      <name>Aaron</name> 
      <age>21</age> 
     </user> 
     <user> 
      <name>Allison</name> 
      <age>22</age> 
     </user> 
    </users> 
</customer> 

<!-- XML File B --> 
<customer> 
    <name>Big Co</name> 
    <users> 
     <user> 
      <name>Bob</name> 
      <age>30</age> 
     </user> 
     <user> 
      <name>Barry</name> 
      <age>31</age> 
     </user> 
     <user> 
      <name>Bill</name> 
      <age>32</age> 
     </user> 
    </users> 
</customer> 

<!-- XML File C --> 
<customer> 
    <name>Cool Co</name> 
    <users> 
     <user> 
      <name>Carl</name> 
      <age>40</age> 
     </user> 
     <user> 
      <name>Craig</name> 
      <age>41</age> 
     </user> 
     <user> 
      <name>Chris</name> 
      <age>42</age> 
     </user> 
    </users> 
</customer> 

我有一个 “根” 的文件,看起来像:

<?xml version='1.0' encoding='utf-8' ?> 
<customers> 
</customers> 

我想将三个A,B和C文件结合起来的nDer根文档,这样我的最终产品将是:

<?xml version='1.0' encoding='utf-8' ?> 
<customers> 
    <!-- XML File A --> 
    <customer> 
     <name>ACME</name> 
     <users> 
      <user> 
       <name>Alex</name> 
       <age>20</age> 
      </user> 
      <user> 
       <name>Aaron</name> 
       <age>21</age> 
      </user> 
      <user> 
       <name>Allison</name> 
       <age>22</age> 
      </user> 
     </users> 
    </customer> 

    <!-- XML File B --> 
    <customer> 
     <name>Big Co</name> 
     <users> 
      <user> 
       <name>Bob</name> 
       <age>30</age> 
      </user> 
      <user> 
       <name>Barry</name> 
       <age>31</age> 
      </user> 
      <user> 
       <name>Bill</name> 
       <age>32</age> 
      </user> 
     </users> 
    </customer> 

    <!-- XML File C --> 
    <customer> 
     <name>Cool Co</name> 
     <users> 
      <user> 
       <name>Carl</name> 
       <age>40</age> 
      </user> 
      <user> 
       <name>Craig</name> 
       <age>41</age> 
      </user> 
      <user> 
       <name>Chris</name> 
       <age>42</age> 
      </user> 
     </users> 
    </customer> 
</customers> 

我一直在寻找的appendChild和ImportNode,但我不断收到各种错误。有一件事是,在我的根文档中,单个空节点customers被列为System.String类型而不是XmlNode,所以我不能附加任何子节点。看到这个PowerShell的片段:

$doc = New-Object System.Xml.XmlDocument 
$doc.LoadXml("<?xml version='1.0' encoding='utf-8' ?><customers></customers>") 
$doc.customers.GetType() 

IsPublic IsSerial Name BaseType 
-------- -------- ---- -------- 
True  True  String System.Object 

甚至不管那么多了,虽然,因为如果我尝试导入的节点,我得到一个错误Cannot import nodes of type 'Document'.

$docA = New-Object System.Xml.XmlDocument 
$docA.LoadXml("<customer><name>ACME</name><users><user><name>Alex</name><age>20</age></user><user><name>Aaron</name><age>21</age></user><user><name>Allison</name><age>22</age></user></users></customer>") 
$docAImported = $doc.ImportNode($docA, $true) 

Exception calling "ImportNode" with "2" argument(s): "Cannot import nodes of type 'Document'." 
At line:1 char:32 
+ $docAImported = $doc.ImportNode <<<< ($docA, $true) 
    + CategoryInfo   : NotSpecified: (:) [], MethodInvocationException 
    + FullyQualifiedErrorId : DotNetMethodException 

任何帮助将不胜感激。谢谢!

回答

2

做第一个获取客户端节点如下:

$customersNode = $doc.ChildNodes | ? { $_.name -eq "customers" } 

现在你可以调用的appendChild在$ customersNode文件A,B和C

但是导入文件A,B和C你几乎没有错。使用documentElement属性象下面这样:

$docAImported = $doc.ImportNode($docA.DocumentElement, $true) 
+0

谢谢,'$ doc.ChildNodes | ? {$ _。name -eq“customers”}'部分是我所需要的。我给出的例子并不完全是我正在使用的XML的一对一,但它足以解决问题。 – romellem

3

如果你需要从XML树节点工作,我建议你通过SelectSingleNode()SelectNodes()选择节点(S)和XPath expression

[xml]$doc = "<?xml version='1.0' encoding='utf-8' ?><customers></customers>" 
$root = $doc.SelectSingleNode('/customers') 

然后你可以importappend从您的其他XML文件是这样的节点:

Get-ChildItem '*.xml' | % { 
    [xml]$xml = Get-Content $_.FullName 
    $node = $xml.SelectSingleNode('/customer') 
    $importedNode = $doc.ImportNode($node, $true) 
    $root.AppendChild($importedNode) 
} 

$doc.Save('C:\path\to\customers.xml')