01.创建数组
- 最标准的使用
@()
,$data = @('Zero','One','Two','Three')
- 使用逗号分隔列表 ,
$data = 'Zero','One','Two','Three'
- 创建连续的数字数组,
$nums=1..5
PS> $data = @('Zero','One','Two','Three')
PS> $data.count
4
PS> $data
Zero
One
Two
Three
也可以是多行
$data = @(
'Zero'
'One'
'Two'
'Three'
)
还有可以用 Write-Output 创建数组
创建字符串数组,默认是可以不加引号的
$data = Write-Output Zero One Two Three
创建指定大小的数组
数组的大小创建后就不能改变
$data = [Object[]]::new(4)
$data.count
4
创建时初始化数组
int默认是初始化为0的
PS> [int[]]::new(4)
0
0
0
0
也可以用数组乘法
PS> $data = @(0) * 4
PS> $data
0
0
0
0
乘法的好处是可以指定任意值
PS> $data = @(255) * 4
PS> $data
255
255
255
255
嵌套数组
$data = @(@(1,2,3),@(4,5,6),@(7,8,9))
$data2 = @(
@(1,2,3),
@(4,5,6),
@(7,8,9)
)
PS> $outside = 0
PS> $inside = 2
PS> $data[$outside][$inside]
3
02.访问数组元素
最基本的通过下标访问
PS> $data = 'Zero','One','Two','Three'
PS> $data[0]
Zero
还有以下几个技巧
- 使用多个索引访问多个元素
$data[0,2,3]
- 使用内置的
..
符号返回连续数字索引的内容$data[1..3]
- 方式2也可以反向
$data[3..1]
- 负数索引表示从尾部开始便宜
$data[-1]
PS> $data[0,2,3]
Zero
Two
Three
PS> $data[3,0,3]
Three
Zero
Three
PS> $data[1..3]
One
Two
Three
PS> $data[3..1]
Three
Two
One
PS> $data[-1]
Three
PS> $a = 1,2,3,4,5,6,7,8
PS> $a[2..-1]
3
2
1
8
PS> $a[2,1,0,-1]
3
2
1
8
03.数组越界
很多编程语言中,数组越界会报错,但是powershell中不会返回任何信息
PS> $null -eq $data[9000]
True
空数组不能索引
PS> $empty = $null
PS> $empty[0]
Error: Cannot index into a null array.
04.数组长度
注意$null
的长度返回0
PS> $data.count
4
PS> $date = Get-Date
PS> $date.count
1
PS> $null.count
0
$data.GetUpperBound(0)
获取数组的边界索引
PS> $data.GetUpperBound(0)
3
PS> $data[ $data.GetUpperBound(0) ]
Three
05.修改数组元素
$data[2] = 'dos'
$data[3] = 'tres'
06.遍历数组元素
001. 管道
PSItem
和$_
是同一个对象
PS> $data = 'Zero','One','Two','Three'
PS> $data | ForEach-Object {"Item: [$PSItem]"}
Item: [Zero]
Item: [One]
Item: [Two]
Item: [Three]
002.ForEach循环
foreach ( $node in $data )
{
"Item: [$node]"
}
003. ForEach方法
PS> $data.foreach({"Item [$PSItem]"})
Item [Zero]
Item [One]
Item [Two]
Item [Three]
004.for 循环
for ( $index = 0; $index -lt $data.count; $index++)
{
"Item: [{0}]" -f $data[$index]
}
005.switch 循环
$data = 'Zero','One','Two','Three'
switch( $data )
{
'One'
{
'Tock'
}
'Three'
{
'Tock'
}
Default
{
'Tick'
}
}
Tick
Tock
Tick
Tock
07.对象数组
$data = @(
[pscustomobject]@{FirstName='Kevin';LastName='Marquette'}
[pscustomobject]@{FirstName='John'; LastName='Doe'}
)
$data[0].FirstName
# 可以用where对数组的属性进行过滤
$data | Where-Object {$_.FirstName -eq 'Kevin'}
$data.Where({$_.FirstName -eq 'Kevin'})
很多cmdlets命令就是返回的对象组成的数组
$processList = Get-Process
08.数组相关的操作符
001.join
PS> $data = @(1,2,3,4)
PS> $data -join '-'
1-2-3-4
PS> $data -join ','
1,2,3,4
没有分隔符的情况下拼接数组,有两种
PS> $data = @(1,2,3,4)
PS> $data -join $null
1234
PS> $data = @(1,2,3,4)
PS> -join $data
1234
002.-replace -split
split是和join相反的操作符
PS> $data = @('ATX-SQL-01','ATX-SQL-02','ATX-SQL-03')
PS> $data -replace 'ATX','LAX'
LAX-SQL-01
LAX-SQL-02
LAX-SQL-03
003.-contains -in 数组包含
PS> $data = @('red','green','blue')
PS> $data -contains 'green'
True
-in和-contains 使用的时候,只是反过来用
PS> $data = @('red','green','blue')
PS> 'green' -in $data
True
如果列表很大,开销就很大,一般我们会用正则来判断
PS> $data = @('red','green','blue')
PS> $pattern = "^({0})$" -f ($data -join '|')
PS> $pattern
^(red|green|blue)$
PS> 'green' -match $pattern
True
004.-eq -ne
数组和eq相等判断的时候,会返回匹配的项目
PS> $data = @('red','green','blue')
PS> $data -eq 'green'
green
ne则是返回不匹配的项目
PS> $data = @('red','green','blue')
PS> $data -ne 'green'
red
blue
如果是在if ()语句中时使用时,则是返回布尔值Boolean
$data = @('red','green','blue')
if ( $data -eq 'green' )
{
'Green was found'
}
if ( $data -ne 'green' )
{
'And green was not found'
}
005. -match
匹配数组中的每个元素
PS> $servers = @(
'LAX-SQL-01'
'LAX-API-01'
'ATX-SQL-01'
'ATX-API-01'
)
PS> $servers -match 'SQL'
LAX-SQL-01
ATX-SQL-01
每次执行match $matches
变量就会被填充值
我们也可以用 Select-String
达到类似的效果
$servers | Select-String SQL
006.乘数组
PS> $data = @('red','green','blue')
PS> $data * 3
red
green
blue
red
green
blue
red
green
blue
09.判断数组为空
之前我们了解到 -eq 会对左边数组中所有项目进行匹配,所以如果里面有一个$null
值,就会返回True
所以我们要把$null
放在eq操作符左边,这样就不会对左边进行遍历
if ( $null -eq $array )
{
'Array actually is $null'
}
但是$null
数组和空数组不同,如果你确定变量是一个数组,可以用count属性判断
有一个陷阱就是对象也可以有count属性
if ( $array.count -gt 0 )
{
"Array isn't empty"
}
下面是绝对安全的数组非空判断方法
if ( $null -ne $array -and @($array).count -gt 0 )
{
"Array isn't empty"
}
10.数组增加元素
数组每次增加元素都是创建一个新数组,所以效率比较低
+和+=语法
$first = @(
'Zero'
'One'
)
$second = @(
'Two'
'Three'
)
PS> $first + $second
Zero
One
Two
Three
$data = @(
'Zero'
'One'
'Two'
'Three'
)
$data += 'four'
11.数组类型
001.强类型数组
强类型数组,只能包含指定类型的值
PS> [int[]] $numbers = 1,2,3
PS> [int[]] $numbers2 = 'one','two','three'
ERROR: Cannot convert value "one" to type "System.Int32". Input string was not in a correct format."
PS> [string[]] $strings = 'one','two','three'
002.ArrayList
数组中无法添加项目,每次拼接的时候,都是重新创建一个新的数组。
所以当我们需要速度更快的数组的时候,可以用ArrayList
这属于是powershell调用 .NET
指定[void]类型是为了避免返回代码,不加的话会返回0
$myarray = [System.Collections.ArrayList]::new()
[void]$myArray.Add('Value')
如果数组中唯一的数据是字符串,可以考虑使用StringBuilder
003.泛型列表
C#是支持泛型的
$mylist = [System.Collections.Generic.List[string]]::new()
$mylist = [System.Collections.Generic.List[int]]::new()
我们也可以将powershell中的数组强制类型转换
$mylist = [System.Collections.Generic.List[int]]@(1,2,3)
就类似于C++,可以声明命名空间来,简化语法
using namespace 声明需要写在脚本的第一行。
using namespace System.Collections.Generic
$myList = [List[int]]@(1,2,3)
同样有Add方法,不同点在于没有返回值,所以不用加void了
$myList.Add(10)
可以像其他数组一样使用
PS> $myList[-1]
10
还有一些常用的方法,比如remove
using namespace System.Collections.Generic
$myList = [List[string]]@('Zero','One','Two','Three')
[void]$myList.Remove("Two")
Zero
One
Three
当使用值类型的时候,remove会删除列表中第一个匹配的值。
004. 对象数组 List[PSObject]
当有一个不知道什么类型的对象的列表时,可以使用[List[PSObject]]
$list = [List[PSObject]]::new()
12.数组取消枚举
powershell中的数组进入管道后就会对每个元素进行枚举,
解决这个问题可以用 Write-Output -NoEnumerate
PS> $data = @('red','green','blue')
PS> $data | Get-Member
TypeName: System.String
...
PS> Write-Output -NoEnumerate $data | Get-Member
TypeName: System.Object[]
...
还有一种比较偏门的方法,就是在数组变量前加一个逗号,不建议使用
PS> ,$data | Get-Member
TypeName: System.Object[]
...