Powershell将一个长数组分成长度为N的数组数组?

时间:2012-12-15 00:50:09

标签: powershell

例如,给定一个列表1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8和一个数字4,它返回一个长度为4的列表列表,即 (1, 2, 3, 4), (5, 6, 7, 8), (1, 2, 3, 4), (5, 6, 7, 8)

基本上我想在Powershell中实现以下Python代码。

s = 1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8
z = zip(*[iter(s)]*4)  # Here N is 4
# z is (1, 2, 3, 4), (5, 6, 7, 8), (1, 2, 3, 4), (5, 6, 7, 8)

以下脚本返回17而不是5。

$a = 1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8,0
$b = 0..($a.Length / 4) | % {  @($a[($_*4)..($_*4 + 4 - 1)]) } 
$b.Length

7 个答案:

答案 0 :(得分:18)

这有点旧,但我想我会把我用来将数组拆分成块的方法。您可以将Group-Object与构造属性一起使用:

$bigList = 1..1000

$counter = [pscustomobject] @{ Value = 0 }
$groupSize = 100

$groups = $bigList | Group-Object -Property { [math]::Floor($counter.Value++ / $groupSize) }

$groups将是GroupInfo对象的集合;在这种情况下,每个组将有正好100个元素(可以$groups[0].Group$groups[1].Group访问,等等。)我使用对象属性作为计数器,以避免-Property脚本块中的作用域问题,因为简单的$i++不会写回原始变量。或者,您可以使用$script:counter = 0$script:counter++,并在没有自定义对象的情况下获得相同的效果。

答案 1 :(得分:9)

在2009年PowerShell Split-Every Function

写了这个

可能可以改进。

Function Split-Every($list, $count=4) {
    $aggregateList = @()

    $blocks = [Math]::Floor($list.Count / $count)
    $leftOver = $list.Count % $count
    for($i=0; $i -lt $blocks; $i++) {
        $end = $count * ($i + 1) - 1

        $aggregateList += @(,$list[$start..$end])
        $start = $end + 1
    }    
    if($leftOver -gt 0) {
        $aggregateList += @(,$list[$start..($end+$leftOver)])
    }

    $aggregateList    
}

$s = 1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8

$r = Split-Every $s 4

$r[0]
""
$r[1]
""
$r[2]
""
$r[3]

答案 2 :(得分:6)

PS> $a = 1..16
PS> $z=for($i=0; $i -lt $a.length; $i+=4){ ,($a[$i]..$a[$i+3])}
PS> $z.count
4    

PS> $z[0]
1
2
3
4

PS> $z[1]
5
6
7
8

PS> $z[2]
9
10
11
12

PS> $z[3]
13
14
15
16

答案 3 :(得分:1)

$a = 1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8,0
$b = 0..([Math]::ceiling($a.Length / 4) - 1) | 
    % {  @(, $a[($_*4)..($_*4 + 4 - 1)]) } 

不知道为什么我必须在(之后加上逗号。

答案 4 :(得分:1)

Clear-Host

$s = 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18

$count = $s.Length

$split = $count/2

$split --

$b = $s[0..$split]

$split ++

$a = $s[$split..$count]

write-host "first array"

$b

write-host "next array"

$a

#clean up
Get-Variable -Exclude PWD,*Preference | Remove-Variable -EA 0

答案 5 :(得分:1)

@Shay Levy答案:如果你将a的值改为1..15那么你的解决方案就不再适用了(Peter Reavy评论)

所以这对我有用:

$a = 1..15
$z=for($i=0; $i -lt $a.length; $i+=4){if ($a.length -gt ($i+3)) { ,($a[$i]..$a[$i+3])} else { ,($a[$i]..$a[-1])}}
$z.count

答案 6 :(得分:0)

使用 select 提供解决方案。不用担心$list.Count能不能被$chunkSize整除。

function DivideList {
    param(
        [object[]]$list,
        [int]$chunkSize
    )

    for ($i = 0; $i -lt $list.Count; $i += $chunkSize) {
        , ($list | select -Skip $i -First $chunkSize)
    }
}

DivideList -list @(1..17) -chunkSize 4 | foreach { $_ -join ',' }

输出:

1,2,3,4
5,6,7,8
9,10,11,12
13,14,15,16
17