Powershell - 并非所有XML属性都传递给管道

use*_*909 5 xml powershell

我正在编写一个相当实质的Posh模块作为一组RESTful Web服务的CLI前端.我有一个奇怪的问题,我能够表征,但没有找到一个干净的解决方案.

我并不总是知道期望什么属性,并且一些节点可能具有某个属性,其他节点可能没有该属性.我的问题有时(经常阅读)如果一个属性不存在于所有节点中,它将不会传递给任何节点的管道.它似乎与XML中第一个节点中的属性有关.

对于一个非常简单的示例,请考虑这是XML文件example.xml的内容:

<members>
    <member>
        <name>Joe</name>
        <rank>Grand Poobah</rank>
        <serialnumber>1234</serialnumber>
    </member>
    <member>
        <name>Fred</name>
        <serialnumber>1234</serialnumber>
    </member>
 </members>
Run Code Online (Sandbox Code Playgroud)

我导入它并将其分配给xml铸造变量:

[xml]$stuff = Get-Content .\example.xml
Run Code Online (Sandbox Code Playgroud)

然后我把它管道到格式表(或out-gridview,export-csv等)

$stuff.members.member | ft -AutoSize

name  rank          serialnumber
----  ----         ------------
Joe   Grand Poobah  1234
Fred                1234
Run Code Online (Sandbox Code Playgroud)

效果很好.

现在,如果FIRST节点没有rank属性:

<members>
    <member>
        <name>Fred</name>
        <serialnumber>1234</serialnumber>
    </member>
    <member>
        <name>Joe</name>
        <rank>Grand Poobah</rank>
        <serialnumber>1234</serialnumber>
    </member>
 </members>
Run Code Online (Sandbox Code Playgroud)

现在,当我执行相同的导入并将其传递给管道时,"rank"属性不会传递给管道.

$stuff.members.member | ft -AutoSize

name   serialnumber
----   ------------
Fred   1234
Joe    1234
Run Code Online (Sandbox Code Playgroud)

我可以通过显式命名属性来强制它:

$stuff.members.member | select-object -Property name, rank, serialnumber | ft

name  rank          serialnumber
----  ----          ------------
Fred                1234
Joe   Grand Poobah  1234
Run Code Online (Sandbox Code Playgroud)

明确命名属性是一个错误的配方,因为可以在没有警告的情况下添加属性.如果可能的话,我希望保持代码灵活处理新属性.

我以为我可以使用"select-object -property*",但是添加了很多我不想要的powershell插入的XML东西,并且仍然没有管道rank属性.

$stuff.members.member | select-object -Property *| ft -AutoSize
name serialnumber LocalName NamespaceURI Prefix NodeType ParentNode OwnerDocument IsEmpty Attributes
---- ------------ --------- ------------ ------ -------- ---------- ------------- ------- ----------
Fred 1234         member                         Element members    #document       False {}
Joe  1234         member                         Element members    #document       False {}
Run Code Online (Sandbox Code Playgroud)

有谁熟悉处理这种情况的好方法?我没有在Google搜索上找到任何好的线索.

Tom*_*lak 5

XML元素肯定是"传递给管道".你正在看的是一个特点format-table:它只查看列表中的第一个对象,以确定要打印的列.

这主要是出于性能原因.否则format-table必须迭代列表两次 - 一次确定要输出的属性,第二次实际输出它们.另外,列表中的项目通常是统一的.

在所有其他情况下,规则"明确比隐含更好"适用:

$stuff.members.member | ft name, rank, serialnumber -AutoSize
Run Code Online (Sandbox Code Playgroud)