用背景信息补充 Ansgar Wiecher的优雅回答:
[pscustomobject]
让我们定义一个示例输入,该输入模拟通过以下方式转换为 PowerShell实例的单个嵌套对象ConvertFrom-Json
:
$objFromJson = [pscustomobject] @{
Object1 = [pscustomobject] @{key1='o11'; key2='o12'; key3='o13'; key4='o14'}
Object2 = [pscustomobject] @{key1='o21'; key2='o22'; key3='o23'; key4='o24'}
Object3 = [pscustomobject] @{key1='o31'; key2='o32'; key3='o33'; key4='o34'}
Object4 = [pscustomobject] @{key1='o41'; key2='o42'; key3='o43'; key4='o44'}
Object5 = [pscustomobject] @{key1='o51'; key2='o52'; key3='o53'; key4='o54'}
}
输出$objFromJson
给出格式如问题中的输出。
为什么这会导致问题中显示的输出格式?
对于诸如 没有为它们定义显式格式定义的类型[pscustomobject]
(通过文件并隐*.ps1xml
式加载到会话中或显式通过),PowerShell 会根据类型的属性数量Update-FormatData
决定要使用的默认格式:
- 具有最多 4 个属性的类型隐式使用
Format-Table
- 具有 5 个或更多属性的类型隐式使用
Format-List
问题中的样本输入可能被删节了;如果真的只有 4 个属性,就会产生表格显示。
属性本身是通过调用它们的值来呈现的.PSObject.ToString()
,这通常与您在双引号字符串中引用对象时得到的表示相同,除了后者始终使用文化不变格式,而.ToString()
将尊重当前文化, 如果类型支持它。
在实例的情况[pscustomobject]
下,这会导致类似于哈希表文字的表示,但不是一个(有关背景信息,请参阅此答案);例如:
PS> $objFromJson.Object1.PSObject.ToString()
@{key1=o11; key2=o12; key3=o13; key4=o14}
根据需要重塑数据:
无法Format-Table
直接使用格式化 cmdlet来产生所需的输出 - 数据必须首先重新整形:
具体来说,对象的属性$objFromJson
必须重新塑造成自定义对象的集合:
$objFromJson
PowerShell 向所有对象添加(除其他外)隐藏.PSObject
属性,其自身的.Properties
属性包含所有对象的属性定义(名称、值、附加元数据,如属性类型等)的集合,有助于提取的属性; 例如:
PS> $objFromJson.Object1.PSObject.Properties
MemberType : NoteProperty
IsSettable : True
IsGettable : True
Value : o11
TypeNameOfValue : System.String
Name : key1
IsInstance : True
# ... remaining properties
$objFromJson
输出'属性定义的集合并仅提取定义'Name
和Value
属性是朝着正确方向迈出的一步:
PS> $objFromJson.PSObject.Properties | Select-Object Name, Value
Name Value
---- -----
Object1 @{key1=o11; key2=o12; key3=o13; key4=o14}
Object2 @{key1=o21; key2=o22; key3=o23; key4=o24}
Object3 @{key1=o31; key2=o32; key3=o33; key4=o34}
Object4 @{key1=o41; key2=o42; key3=o43; key4=o44}
Object5 @{key1=o51; key2=o52; key3=o53; key4=o54}
但是,我们必须使属性的.Value
属性直接输出对象的属性才能获得具有属性个体值的输出。
Ansgar 的优雅回答演示了如何在单个管道中做到这一点。
让我用从它派生的可重用辅助函数来补充它:
function ConvertTo-Collection($InputObject) {
foreach ($obj in $InputObject) {
foreach ($prop in $obj.PSObject.Properties) {
$prop.Value | Select-Object @{ n='Name'; e={ $prop.Name }}, *
}
}
}
有了该功能,现在可以实现所需的输出:
ConvertTo-Collection $objFromJson | Format-Table
排除特定属性,例如key3
:
ConvertTo-Collection $objFromJson | Select-Object -ExcludeProperty key3 |
Format-Table