如果我没有列出属性,则Compare-Object无法正常工作

时间:2013-08-15 19:07:17

标签: powershell powershell-v2.0

我想要比较两个Excel电子表格:

$OleDbAdapter = New-Object System.Data.OleDb.OleDbDataAdapter “Select * from [Report$]“,”Provider=Microsoft.ACE.OLEDB.12.0;Data Source=S:\FIS-BIC Reporting\Report Output Files\Product-Marketing\TEST_XI\ECM - Pipeline by LOB_04182013_040544.xls;Extended Properties=”"Excel 12.0 Xml;HDR=YES”";”
$RowsReturned = $OleDbAdapter.Fill($DataTable)

$OleDbAdapter2 = New-Object System.Data.OleDb.OleDbDataAdapter “Select * from [Report$]“,”Provider=Microsoft.ACE.OLEDB.12.0;Data Source=S:\FIS-BIC Reporting\Report Output Files\Product-Marketing\ECM - Pipeline by LOB_04182013_074004.xls;Extended Properties=”"Excel 12.0 Xml;HDR=YES”";”
$RowsReturned2 = $OleDbAdapter2.Fill($DataTable2)

Compare-Object $DataTable $DataTable2 

它什么都不返回。我知道在第6栏中,它们是不同的。如果我指定“-property F6”,它确实会返回差异。除非我指定属性,否则任何想法都没有?列数可以变化(虽然对于比较中的每个文件都是相同的),因此具体指定属性将不起作用。

2 个答案:

答案 0 :(得分:23)

如果您未指定 -Property 参数, Compare-Object 不会比较所有属性,则会比较调用 .ToString的结果两个对象上的()方法。因此,Compare-Object $DataTable $DataTable2 $ DataTable1.ToString() $ DataTable1.ToString()进行比较。 .ToString()方法在DataTable对象上调用时返回一个空字符串,因此报告没有区别。

例如:

$file1 = Get-Item somefilename
$file1 = Get-Item anotherfilename
Compare-Object $file1 $file2

这将返回两个文件的完整路径之间的差异,如下所示:

InputObject              SideIndicator
-----------              -------------
<path>\anotherfilename   =>
<path>\somefilename      <=

这是因为在FileInfo对象上调用 .ToString()会返回其 FullName 属性,因此您要比较文件的完整路径名。

虽然 -Property 参数接受多个属性,但列出所有属性不是解决方案。除了非常乏味,它不会给你你想要的结果。如果列出多个属性,比较对象会比较所有属性的组合,如果列出的属性中的任何一个属性不同,则会返回显示所有列出的属性的结果属性(两者相同,属性不同)作为单一差异。

您需要做的是迭代属性列表,并为每个属性调用 Compare-Object 一次:

$properties = ($DataTable | Get-Member -MemberType Property | Select-Object -ExpandProperty Name)
foreach ($property in $properties) {
  Compare-Object $DataTable $DataTable2 -Property "$property" | Format-Table -AutoSize
}
  • 在大多数情况下,在比较两个对象的所有属性时,您需要使用Get-Member -MemberType Properties,以便涵盖所有属性类型。但是,如果您要比较DataTable对象,最好使用Get-Member -MemberType Property,这样您只能比较与数据字段对应的属性,而不是与数据无关的DataTable对象的其他属性。

  • 这是假设列数与您所说的相同,或者至少 $ DataTable2 中的列数不超过列数的 $数据表即可。

    如果您无法可靠地假设,通过将($DataTable | Get-Member -MemberType Property).Count($DataTable2 | Get-Member -MemberType Property).Count进行比较并使用其中的属性,从任何一个列中获取更多列的 $ properties 数组更大。

  • 使用格式表非常重要,不仅仅是让事情看起来很漂亮。如果列出多个相同类型的对象(在本例中为数组),PowerShell会记住第一个对象的格式,并将其用于所有后续对象,除非您明确指定格式。由于第一列的名称对于每个属性(即电子表格中的每一列)将是不同的,因此除了遇到的第一个差异之外,第一列将为空。

    -AutoSize开关是可选的。那个 只是为了让事情看起来很漂亮。但是您必须将结果传递给格式化过滤器。如果您愿意,也可以使用格式列表

答案 1 :(得分:2)

Adi Inbar's helpful answer包含有关Compare-Object如何运作的良好背景信息。

但是, 是一种使用-Property来一般比较所有列值的方法 - 假设两个输入表具有相同的列结构,或者这些表只能通过第一个表的列进行比较:

# The original collection.
$coll1 = [pscustomobject] @{ one = '1a'; two = '2a'; three = '3a' },
         [pscustomobject] @{ one = "1b"; two = "2b"; three = '3b' }

# The other collection to compare the original to.
# Note the difference in the 2nd object in column 'two'
$coll2 = [pscustomobject] @{ one = '1a'; two = '2a'; three = '3a' },
         [pscustomobject] @{ one = "1b"; two = "2b!"; three = '3b' }

# PSv3+: Get the array of all property names to compare 
#        from the original collection.
# Note: 
#      * The assumption is that both collections have the same set of 
#        properties (or that the collections should only be compared by
#        the *first* collection's properties).
#      * In PSv2-, use the following instead:
#         $propsToCompare = $coll1[0].psobject.properties | % { $_.name }
$propsToCompare = $coll1[0].psobject.properties.name

# Compare the 2 collections by all property values.
# -PassThru means that any input object in which a difference is found
# is passed through as-is.
Compare-Object $coll1 $coll2 -Property $propsToCompare -PassThru

以上产量:

one two three SideIndicator
--- --- ----- -------------
1b  2b! 3b    =>           
1b  2b  3b    <=           

请注意=>如何告诉您所选对象对于右侧侧是独占的,反之亦然<=

需要注意的是 Compare-Object ,因为它无法对排序 的输入数据做出假设,因此必须读取并比较两个输入集合完整

使用排序输入,您可以使用-SyncWindow <Int32>加快速度,但这需要事先了解最多多少项在两个输入集合之间可能有所不同,如果-SyncWindow值太小,将报告虚假差异。