为什么这个Powershell函数有时返回一个列表而其他时候返回一个XmlElement?

时间:2014-09-19 22:33:37

标签: xml powershell

由于某种原因,如果我的XML文件只包含一个计算机节点,则以下Read-TestControllerXml powershell函数返回System.Xml.XmlElement类型的对象,但它返回一个列表(这是我一直期望的)如果有的话是多个计算机节点。

XML文件的示例如下:

<test_controller>
    <defaults>
        <nunit path='C:\Program Files (x86)\NUnit 2.6.2\bin\nunit-console.exe' />
        <nunit_results local_path='C:\Test_Results' remote_path='C:\Test_Results' />
        <powershell path='C:\windows\system32\WindowsPowerShell\v1.0\powershell.exe' />
        <psexec path='C:\PSTools\PsExec.exe' />
    </defaults>

    <platform os='windows' version='2012' cpu='x64'>
      <computer hostname='VM-2012R2-03' alias='standalone-01' configuration='standalone' image='VM-2012R2-03:Clean' />
    </platform>
</test_controller>

这是powershell函数:

####################################################################
# Parses the TestController.xml file and returns a list of Computer XML objects.
#
# Syntax:
#     Read-TestControllerXml [-TestControllerXml] <path to TestController.xml> [<CommonParameters>]
#
# Returns:  An array of [Xml] nodes for each of the <computer> tags in TestController.xml.
####################################################################
function Read-TestControllerXml
{
    [CmdletBinding()]
    Param
    (
        [parameter(Mandatory=$true)]
        [string] $TestControllerXml
    )

    $ErrorActionPreference = "Stop"

    # Read the TestController XML configuration file
    $TestController = [xml] (Get-Content $TestControllerXml)

    Write-Verbose "Selecting computer nodes..."
    $computerNodes = $TestController.SelectNodes("//test_controller/platform/computer")
    Write-Verbose "Selecting default nodes..."
    $defaultNodes = $TestController.SelectSingleNode("//test_controller/defaults")

    $computers = @()

    foreach ($computerNode in $computerNodes)
    {
        Write-Verbose "***** Before adding default attributes *****"
        Write-Verbose "$($computerNode.OuterXml)"

        # Add default attributes to the node
        foreach ($dnode in $defaultNodes.ChildNodes)
        {
            if ($dnode.NodeType -eq "Comment") { continue } # Skip comments.

            # Search for the node items in the defaults section
            $cnode = $computerNode

            if ($dnode.Name -ne $computerNode.Name)
            {
                Write-Verbose "Selecting '$($dnode.Name)' nodes..."
                $cnode = $computerNode.SelectSingleNode($dnode.Name)
            }

            # Append the default nodes that do not exist in the computer specific sections  
            if ($cnode -eq $null)
            {
                Write-Verbose "*** cnode != null  cnode.OuterXml is: '$($cnode.OuterXml)'"
                Write-Verbose "Appending default '$($dnode.Name)' node that don't exist in the computer nodes..."
                $cnode=$computerNode.AppendChild($TestController.ImportNode($dnode, $true))
            }
            # Append the default attributes that do not exist in the computer specific sections  
            else
            {
                Write-Verbose "Appending default attributes that don't exist in computer nodes..."
                foreach ($attribute in $dnode.Attributes)
                {
                    if (-not $cnode.HasAttribute($attribute.Name))
                    {
                        Write-Verbose "Adding attribute '$($attribute.Name)=$($attribute.Value)' to the computer node."
                         $cnode.SetAttribute($attribute.Name, $attribute.Value)
                    }
                }
            }
        }

        Write-Verbose "***** After adding default attributes *****"
        Write-Verbose "$($computerNode.OuterXml)"
        Write-Verbose "************************************************************"

        $computers += $computerNode
    }

    return $computers
}

2 个答案:

答案 0 :(得分:1)

我拿了你的例子函数和样本数据。运行该函数我得到你描述的输出。

PS C:\Users\Matt> (Read-TestControllerXml C:\temp\xml.txt).GetType().FullName
System.Xml.XmlElement

然后我将一个计算机节点添加到示例文件并再次运行该函数。

PS C:\Users\Matt> (Read-TestControllerXml C:\temp\xml.txt).GetType().FullName
System.Object[]

这是正确的,因为你的函数返回和对象数组。如果您分解项目并检查其类型,它应该是有道理的。

PS C:\Users\Matt> Read-TestControllerXml C:\temp\xml.txt | ForEach-Object{$_.GetType().FullName}
System.Xml.XmlElement
System.Xml.XmlElement

我有两个项目并将它们放入ForEach-Object循环并单独检查其类型。由于您的输出是一个数组,期望有多个对象。总之,只要有多个[System.Xml.XmlElement]数组,您的函数就会返回System.Object[]

出于好奇的缘故

我试图强制$ computers成为像这个$computers = [System.Xml.XmlElement[]]@()的xml元素数组。它没有改变输出。多个项目仍作为System.Object[]

输出

答案 1 :(得分:0)

我刚刚发现了这个类似的问题,其解决方案有效: Function not returning expected object

即。变化:

return $computers

为:

return ,$computers