如何使用Import-CliXml反序列化自定义类对象的通用列表?

时间:2016-10-25 02:22:22

标签: xml powershell deserialization generic-list

我正在尝试在PowerShell中学习XML序列化。所以,我正在制作一个简单的联系人列表来序列化和反序列化。我遇到的问题是Contact对象在使用Import-CliXml导入时的行为有所不同。

以下是我现在正在尝试的内容:

class Contact {
[string]$FirstName;
[string]$LastName;
[Int64]$PhoneNumber; }

$bob = new-object Contact
$bob.FirstName = "Bob"
$bob.LastName = "Johnson"
$bob.PhoneNumber = 1235589876

$bill = new-object Contact
$bill.FirstName = "Bill"
$bill.LastName = "Carson"
$bill.PhoneNumber = 1235589875

$ContactList = new-object System.Collections.Generic.List[Contact]
$ContactList.Add($bob)
$ContactList.Add($bill)

[xml]$data = $ContactList | convertto-xml

new-item "C:\Users\Public\Documents\Contacts.xml"
$data | export-clixml "C:\Users\Public\Documents\Contacts.xml"

[xml]$ImportedData = import-clixml "C:\Users\Public\Documents\Contacts.xml"
$ImportedList = new-object System.Collections.Generic.List[Contact]
$ImportedList = $ImportedData.Objects.Object

导入的List[Contact]包含Contact个对象的集合,但我似乎无法从其属性中获取信息。例如,当我尝试以下操作时出现空行:

$ImportedList | foreach-object { write-host $_.FirstName }

我知道对象在那里,因为我从$ImportedList获得了回报:

$ImportedList

Type    Property
----    --------
Contact {FirstName, LastName, PhoneNumber}
Contact {FirstName, LastName, PhoneNumber}

我试图让它返回与$ContactList相同的东西:

$ContactList

FirstName LastName PhoneNumber
--------- -------- -----------
Bob       Johnson  1235589876
Bill      Carson   1235589875

非常感谢任何帮助。

2 个答案:

答案 0 :(得分:1)

您遇到的问题是import-clixml,它不会返回对象类型[contact]$ImportedData.Objects.Object而不是[System.Xml.XmlElement]类型,这就是为什么你在这里得到空行:

$ImportedList | foreach-object { write-host $_.FirstName }

因为管道中的对象没有属性firstname

而不是这两行:

$ImportedList = new-object System.Collections.Generic.List[Contact]
$ImportedList = $ImportedData.Objects.Object

试试这个:

$ImportedList = @()
$ImportedData.Objects.Object | %{$ImportedList += [contact]@{firstname=$_.property[0].'#text';lastname=$_.property[1].'#text';phonenumber=$_.property[2].'#text'}}

答案 1 :(得分:0)

谢谢Tav明确答复。这就是我最终使用的:

$ImportedList = new-object System.Collections.Generic.List[Contact]

$ImportedData.Objects.Object | foreach-object {
$NewContact = new-object Contact
$NewContact.FirstName = $_.Property[0].'#text'
$NewContact.LastName = $_.Property[1].'#text'
$NewContact.PhoneNumber = [Int64]$_Property[2].'#text'
$ImportedList.Add($NewContact) }

<强>更新

我能够使用Where()方法使用属性名称:

$ImportedList = new-object System.Collections.Generic.List[Contact]

$ImportedData.Objects.Object | foreach-object {
$NewContact = new-object Contact
$NewContact.FirstName = $_.Property.Where({ $_.Name -eq "FirstName" }).'#text'
$NewContact.LastName = $_.Property.Where({ $_.Name -eq "LastName" }).'#text'
$NewContact.PhoneNumber = [Int64]$_Property.Where({ $_.Name -eq "PhoneNumber" }).'#text'
$ImportedList.Add($NewContact) }