Powershell 加入/合并数组';性质
我有两个对象,我想合并,但似乎我找不到解决方案Powershell 加入/合并数组';性质,powershell,Powershell,我有两个对象,我想合并,但似乎我找不到解决方案 Messages Name Error -------- ----
Messages Name Error
-------- ---- -----
{\\APPS-EUAUTO1\C$\Users\xautosqa\AppDa... test 1 True
{[APPS-EUAUTO1] [prep] Setting agent op... test 2 False
TestPlan Script TestCase TestData ErrorCount ErrorText DateTime Elapsed
-------- ------ -------- -------- ---------- --------- -------- -------
D:\XHostMach... D:\XHostMach... rt1 1,\a\"" 1 [#ERROR#][AP... 2014-03-28 1... 0:00:18
D:\XHostMach... D:\XHostMach... rt2 1,\a\"" 0 2014-03-28 1... 0:00:08
我试过:
function Join-Object {
Param(
[Parameter(Position=0)]
$First
,
[Parameter(Position=1,ValueFromPipeline=$true)]
$Second
)
BEGIN {
[string[]] $p1 = $First | gm -type Properties | select -expand Name
}
Process {
$Output = $First | Select $p1
foreach($p in $Second | gm -type Properties | Where { $p1 -notcontains $_.Name } | select -expand Name) {
Add-Member -in $Output -type NoteProperty -name $p -value $Second."$p"
}
$Output
}
} # End: Function Join-Object
$TestCases = Join-Object $TxtTestcases $RexTestcases | Select Name, TestPlan, TestCase, Script, TestData, Error, ErrorCount, ErrorText, Messages, DateTime, Elapsed
但第二个对象不存在:
Name TestPlan TestCase Script TestData Error ErrorCount ErrorText Messages DateTime
---- -------- -------- ------ -------- ----- ---------- --------- -------- --------
test 1 True {\\APPS-...
test 2 False {[APPS-E...
我也尝试过:
Function Merge-Testcase {
Param ($TxtTestcase, $RexTestcase)
$Fields = @{
Name = $TxtTestcase.Name
TestPlan = $RexTestcase.TestPlan
TestCase = $RexTestcase.TestCase
Script = $RexTestcase.Script
TestData = $RexTestcase.TestData
IsError = $TxtTestcase.Error
ErrourCount = $RexTestcase.ErrorCount
ErrorText = $RexTestcase.ErrorText
Messages = $TxtTestcase.Messages
DateTime = $RexTestcase.DateTime
Elapsed = $RexTestcase.Elapsed
}
New-Object PSObject -Property $Fields | ConvertTo-Csv -NoTypeInformation
} # End: Function Merge-Testcase
$TestCases = Merge-Testcase -TxtTestcase $TxtTestcases -RexTestcas $RexTestcase
$TestCases | Format-Table
但我明白了
"Script","Name","TestCase","TestPlan","ErrorText","TestData","ErrourCount","IsError","Elapsed","DateTime","Messages"
,,,,,,,,,,
您知道如何连接这两个对象吗?选项1
一种选择是将这两个对象放入一个哈希表中,然后将其转换为PSCustomObject
$Obj1 = (Get-Process)[0]; # Get a process
$Obj2 = (Get-Service)[0]; # Get a service
($Result = [PSCustomObject]@{ Obj1 = $Obj1; Obj2 = $Obj2; });
$Result.Obj1.Name;
$Result.Obj2.Name;
注意:这将创建一个额外的“级别”,您必须深入了解它,因此它并不理想,但会起作用
选项2
第二个选项是迭代“第二个对象”的所有属性,并使用addmember
将它们添加到“第一个对象”,您的示例已经显示了这一点
创建一个名为c:\test\test.txt
的空文件,然后运行以下代码:
# Get a couple of objects (with different property names)
$Object1 = Get-Service -Name Dnscache;
$Object2 = Get-Item c:\test\test.txt;
# Get a list of the properties on both objects
$PropertyList1 = @(Get-Member -InputObject $Object1 -MemberType Properties).Name;
$PropertyList2 = Get-Member -InputObject $Object2 -MemberType Properties | Where-Object -FilterScript { $PropertyList1 -notcontains $PSItem.Name; };
# Add the properties, from the second object, to the first object
foreach ($Property in $PropertyList2) {
Write-Host ('Adding property: {0}' -f $Property.Name);
Add-Member -InputObject $Object1 -Name $Property.Name -MemberType NoteProperty -Value $Object2.$($Property.Name);
}
# Output the object
$Object1 | select *;
输出如下所示:
Mode : -a---
PSChildName : test.txt
PSDrive : C
PSIsContainer : False
PSParentPath : Microsoft.PowerShell.Core\FileSystem::C:\test
PSPath : Microsoft.PowerShell.Core\FileSystem::C:\test\test.txt
PSProvider : Microsoft.PowerShell.Core\FileSystem
Attributes : Archive
CreationTime : 3/11/2014 3:06:43 PM
CreationTimeUtc : 3/11/2014 8:06:43 PM
Directory : C:\test
DirectoryName : C:\test
Exists : True
Extension : .txt
FullName : C:\test\test.txt
IsReadOnly : False
LastAccessTime : 3/11/2014 3:06:43 PM
LastAccessTimeUtc : 3/11/2014 8:06:43 PM
LastWriteTime : 3/11/2014 3:06:29 PM
LastWriteTimeUtc : 3/11/2014 8:06:29 PM
Length : 0
BaseName : test
VersionInfo : File: C:\test\test.txt
InternalName:
OriginalFilename:
FileVersion:
FileDescription:
Product:
ProductVersion:
Debug: False
Patched: False
PreRelease: False
PrivateBuild: False
SpecialBuild: False
Language:
Name : Dnscache
RequiredServices : {nsi, Tdx}
CanPauseAndContinue : False
CanShutdown : False
CanStop : True
DisplayName : DNS Client
DependentServices : {NcaSvc}
MachineName : .
ServiceName : Dnscache
ServicesDependedOn : {nsi, Tdx}
ServiceHandle :
Status : Running
ServiceType : Win32ShareProcess
Site :
Container :
查看服务和文件的属性如何都位于同一个对象上?您的原始帖子中的主要问题似乎是$TxtTestcases和$rextTestCases是分别具有两个对象的数组,但您的Join object函数并不是为了在数组上枚举而编写的。根据我所看到的,如果您这样调用函数,您的函数应该可以正常工作:
连接对象@($TxtTestcases)[0]@($rextTestCases)[0]
这取决于您如何选择数组中的哪些对象被加入,或者如果数组包含不同数量的对象,您该怎么做。此处似乎没有任何要“加入”的公共字段(如果有,可以从中使用“加入”对象函数)。我正在维护一个cmdlet(请参阅:)。
我在这个项目中的任务是尽可能保持语法简单,并获得最佳性能,同时仍然遵守有关管道的PowerShell开发指南。
对于@Dave Wyatt指出的事情:
- “似乎没有任何公共字段”:在SQL
Join
子句中,可以使用ON
参数进行此操作,就像使用Join-Object
cmdlet一样,但如果省略-ON
参数,则会在对象索引上进行连接
- “或者,如果数组包含不同数量的对象,该怎么办”,类似于相关属性上的常用联接,您可以使用包含的代理命令
LeftJoin Object
(别名LeftJoin
),righjoin Object
(别名righjoin
)来控制如何处理不同的表长度和FullJoin对象
(别名FullJoin
)用于此(请参阅)
关于这一具体问题:
$TxtTestcases | Join-Object $RexTestcases | Format-Table *
Messages Name Error TestPlan Script TestCase TestData ErrorCount ErrorText DateTime Elapsed
-------- ---- ----- -------- ------ -------- -------- ---------- --------- -------- -------
{\\APPS-EUAUTO1\C$\Users\xautosqa\AppDa... test 1 True D:\XHostMach... D:\XHostMach... rt1 1,\a\"" 1 [#ERROR#][AP... 2014-03-28 0:00:18
{[APPS-EUAUTO1] [prep] Setting agent op... test 2 False D:\XHostMach... D:\XHostMach... rt2 1,\a\"" 0 2014-03-28 0:00:08
好问题-我发布了一个答案,其中有两个可能的解决方案。我想你会喜欢第二个:)我可以发誓我在四五天前回答了一个和这个一样的问题。事实上,我的答案和特雷弗给你的第二个答案非常相似。您是否尝试过在这里已经回答过的问题中寻找解决方案?@TheMadTechnician是的,我尝试过一些方法,但没有任何效果,我认为因为我使用了数组:$Obj1 | Join$Obj2
会抓住所有各种“属性”类型(例如许多自定义对象最终使用的NoteProperty)是的。我使用了-MemberType属性
,其中包括NoteProperty
。太棒了!我以后将不得不走这条路,这看起来比我使用的GM |?{$\u0.MemberType-Match“Property”}
更干净。谢谢特雷弗@TrevorSullivan当我运行您的代码时,我得到以下错误Add Member:无法将参数绑定到参数'Name',因为它为null。在第12行char:43
中,我不明白why@ionut您正在运行PowerShell 4.0版吗?这就是代码编写和测试的地方。是的,我更改了这个$TestCases=@();对于($i=0;$i-lt$TxtTestcases.length;$i++){$TestCases+=Join Object@($TxtTestcases)[$i]@($rexttestcases)[$i]}
并且它工作得很好,感谢非常好的cmdlet
$TxtTestcases | Join-Object $RexTestcases | Format-Table *
Messages Name Error TestPlan Script TestCase TestData ErrorCount ErrorText DateTime Elapsed
-------- ---- ----- -------- ------ -------- -------- ---------- --------- -------- -------
{\\APPS-EUAUTO1\C$\Users\xautosqa\AppDa... test 1 True D:\XHostMach... D:\XHostMach... rt1 1,\a\"" 1 [#ERROR#][AP... 2014-03-28 0:00:18
{[APPS-EUAUTO1] [prep] Setting agent op... test 2 False D:\XHostMach... D:\XHostMach... rt2 1,\a\"" 0 2014-03-28 0:00:08