.NET Framework ArrayList类.Add方法为什么在PowerShell实现中不起作用?
除非经过更正,否则我认为我的故事的总体寓意可能是:不要假定本机PowerShell方法将与.NET方法相同,并且在PowerShell中尝试.NET方法时要小心。
我寻求的原始解决方案是从函数(作为数组)以用户定义的日期范围作为参数返回日期列表。然后,将引用日期数组以移动和读取以日期戳命名的文件。
我遇到的第一个问题是创建动态数组。我不知道自己在做什么,并且在.Add
数组声明上错误地调用了.NET @()
方法。
当我真正的问题是我做得不好时,我以为我需要找到一个动态数组类型。这使我朝着不同的方向前进,直到很久以后,我才发现应该使用+=
语法将对象添加到PowerShell数组。
无论如何,在我回到如何正确使用PowerShell数组之前,我没有使用其他切线。
然后,我找到了.NET ArrayList
类。好的。现在,我有了一个动态数组对象。我阅读了文档,其中说我应该使用.Add
方法将元素添加到集合中。
然后,我开始寻求更深入的了解,因为我经历了几天的挫折,试图解决问题。
我制作了一个起初似乎可行的实现。它产生了一个日期范围-但它也产生了一些奇怪的行为。我观察到返回的日期很奇怪,例如:
我发现,这是您执行以下操作时获得的结果:
Get-Date 0
ArrayList
首先返回数组元素的索引值列表,然后返回数组值。那根本没有任何意义。我开始研究是否正确地调用了函数,是否遇到了某种可变范围的问题,或者我只是疯了。我现在非常确信,我的沮丧是由于缺乏扎实的初学者引用而引起的,该引用不仅显示了如何进行简单数组实现的几个示例,而且还描述了一些注意事项以及其他解决方案。
然后,让我在这里解释实现数组/集合的三种方式,以及针对我尝试生成的内容的解决方案-即,日期范围内的日期列表。
由于某种原因,我最初认为在Powershell中将元素添加到.NET ArrayList的正确方法是使用.Add方法。这是documented。我仍然不明白为什么该方法不起作用的原因(严重的是,请有人启发我)。通过实验,我发现我可以通过使用
+=
方法将对象添加到ArrayList来获得准确的结果。不要这样这绝对是错误的。它将产生我上面描述的错误:
Function Get-DateRangeList {
[cmdletbinding()]
Param (
[datetime] $startDate,
[datetime] $endDate
)
$datesArray = [System.Collections.ArrayList]@() # Second method
for ($d = $startDate; $d -le $endDate; $d = $d.AddDays(1)) {
if ($d.DayOfWeek -ne 'Sunday') {
$datesArray.Add($d)
}
}
Return $datesArray
}
# Get one week of dates, ending with yesterday's date
$startDate = Get-Date
$endDate = $startDate.AddDays(-1) # Get yesterday's date as last date in range
$startDate = $endDate.AddDays(-7) # Get 7th prior date as first date in range
$datesList = Get-DateRangeList $startDate $endDate
# Loop through the dates
Foreach ($d in $datesList) {
# Do something with each date, e.g., format the date as part of a list
# of date-stamped files to retrieve
$d
}
现在,下面有三个代码示例可以执行。在每个示例中,代码都是相同的。我所做的只是注释/取消注释了相应的实例化行和方法行。
首先,使用本机PowerShell数组对象:
Function Get-DateRangeList {
[cmdletbinding()]
Param (
[datetime] $startDate,
[datetime] $endDate
)
$datesArray = @() # First method
#$datesArray = [System.Collections.ArrayList]@() # Second method
#$datesArray = New-Object System.Collections.Generic.List[System.Object] # Third method
for ($d = $startDate; $d -le $endDate; $d = $d.AddDays(1)) {
if ($d.DayOfWeek -ne 'Sunday') {
$datesArray += $d # First and second method: += is the method to add elements to: Powershell array; or .NET ArrayList (confusing)
#$datesArray.Add($d) # Third method: .Add is the method to add elements to: .NET Generic List
}
}
Return $datesArray
}
# Get one week of dates, ending with yesterday's date
$startDate = Get-Date
$endDate = $startDate.AddDays(-1) # Get yesterday's date as last date in range
$startDate = $endDate.AddDays(-7) # Get 7th prior date as first date in range
$datesList = Get-DateRangeList $startDate $endDate
# Loop through the dates
Foreach ($d in $datesList) {
# Do something with each date, e.g., format the date as part of a list
# of date-stamped files to retrieve
"FileName_{0}.txt" -f $d.ToString("yyyyMMdd")
}
其次,使用.NET Framework
ArrayList
:Function Get-DateRangeList {
[cmdletbinding()]
Param (
[datetime] $startDate,
[datetime] $endDate
)
#$datesArray = @() # First method
$datesArray = [System.Collections.ArrayList]@() # Second method
#$datesArray = New-Object System.Collections.Generic.List[System.Object] # Third method
for ($d = $startDate; $d -le $endDate; $d = $d.AddDays(1)) {
if ($d.DayOfWeek -ne 'Sunday') {
$datesArray += $d # First and second method: += is the method to add elements to: Powershell array; or .NET ArrayList (confusing)
#$datesArray.Add($d) # Third method: .Add is the method to add elements to: .NET Generic List
}
}
Return $datesArray
}
# Get one week of dates, ending with yesterday's date
$startDate = Get-Date
$endDate = $startDate.AddDays(-1) # Get yesterday's date as last date in range
$startDate = $endDate.AddDays(-7) # Get 7th prior date as first date in range
$datesList = Get-DateRangeList $startDate $endDate
# Loop through the dates
Foreach ($d in $datesList) {
# Do something with each date, e.g., format the date as part of a list
# of date-stamped files to retrieve
"FileName_{0}.txt" -f $d.ToString("yyyyMMdd")
}
第三,使用.NET Framework Generic List:
Function Get-DateRangeList {
[cmdletbinding()]
Param (
[datetime] $startDate,
[datetime] $endDate
)
#$datesArray = @() # First method
#$datesArray = [System.Collections.ArrayList]@() # Second method
$datesArray = New-Object System.Collections.Generic.List[System.Object] # Third method
for ($d = $startDate; $d -le $endDate; $d = $d.AddDays(1)) {
if ($d.DayOfWeek -ne 'Sunday') {
#$datesArray += $d # First and second method: += is the method to add elements to: Powershell array; or .NET ArrayList (confusing)
$datesArray.Add($d) # Third method: .Add is the method to add elements to: .NET Generic List
}
}
Return $datesArray
}
# Get one week of dates, ending with yesterday's date
$startDate = Get-Date
$endDate = $startDate.AddDays(-1) # Get yesterday's date as last date in range
$startDate = $endDate.AddDays(-7) # Get 7th prior date as first date in range
$datesList = Get-DateRangeList $startDate $endDate
# Loop through the dates
Foreach ($d in $datesList) {
# Do something with each date, e.g., format the date as part of a list
# of date-stamped files to retrieve
"FileName_{0}.txt" -f $d.ToString("yyyyMMdd")
}
所有这三个工作。为什么您偏爱一个?本机PowerShell数组和.NET Framework的
ArrayList
类都生成非强类型的对象的集合,因此您可以这样做(在Powershell数组实现中):$myArray = @(1, 2, 3, "A", "B", "C")
Powershell阵列对于非常大的阵列而言效率不高。对于非常大的集合,ArrayList是更好的选择。
对于大量相同类型的对象,似乎.NET Framework通用列表是最佳选择。在我的示例中,我想要一个日期列表。每个日期都是相同的数据类型,因此无需混合对象类型。因此,我正在部署的解决方案是上面的第三个工作示例。
感谢Dave Wyatt在2013年Powershell.org上发表的有关PowerShell Performance: The += Operator (and When to Avoid It)的文章。特别是
+=
方法creates a new array object in each pass within a loop,添加新元素,然后销毁旧数组。对于大量收集,这将变得非常低效。我正在发布这些解决方案和讨论,以希望其他初学者可以更轻松地找到我所寻求的答案。
是的-是的-我不遵循某些人的想法,即严格的PowerShell语法礼节。我在函数中使用了
return
语句,因此很明显该函数产生了什么。我更喜欢可读性强的代码,这些代码看起来可能比扩展的要紧。那是我的偏爱,我坚持下去。有关日期列表的更像PowerShell的实现,我请读者引用tidy implementation posted by The Surly Admin。
最佳答案
关于OP的第3段:Collections.arraylist确实可以在Powershell中工作,例如:
# Create arraylist with space for 20 object
$ar = new-object collections.arraylist 20
$ar.add("hello everybody")
$ar.add([datetime]::now)
$ar.add( (gps)[9])
$ar[0] # returns string
$ar[1] # returns datetime
$ar[2] # returns tenth process
$ar.count # returns 3
我认为这是要更仔细地阅读arraylist的MSDN文档。
如果在PS中的arraylist上使用+ =,它将从arraylist中获取元素,并获取新元素并创建一个数组。我相信,这是一种使用户免受偶然发现的.NET复杂性影响的尝试。 (我怀疑PS产品团队的主要用例之一是对.NET尤其是arraylist不太熟悉的用户。您显然不属于该类别。)
我将提到PS和阵列的绊脚石。 PS在某些情况下会自动展开阵列。例如,如果我有一个字符数组,并且想要创建一个字符串(使用String..ctor([char []])重载),则此方法将无效:
# Fails because PS unrolls the array and thinks that each element is a
# different argument to String..ctor
$stringFromCharArray = new-object string $charArray
# Wrap $charArray to get it to work
$stringFromCharArray = new-object string @(,$charArray)
# This also works
$stringFromCharArray = new-object string (,$charArray)
当您沿管道向下传递数组时,也存在类似的问题。如果要使数组沿管道传递(相对于数组元素),则需要先将其包装在另一个数组中。