|
http://hebe852.blog.163.com/blog/static/12072624820108744039615/
熟悉脚本编写的朋友对函数(Function)这个名词一定不会陌生,函数是一系列语句集合而成的代码块,我们可以为其命名。然后只需调用函数名称便能执行函数中的代码,这种方式使得我们无需每次都输入一大段代码来完成一些特定的功能。
在WindowsPowerShell
v2中,函数有了很大的增强。高级函数在执行操作时可以得到与cmdlet类似的操作体验。毕竟对广大使用PowerShell的非开发人员而言,与使
用.NET语言去编写cmdlet相比,使用熟悉的PowerShell语法去编写函数还是很简单的。今天我们主要先熟悉下高级函数的一些概念,在下次的
介绍中我们再来谈谈如何使用高级函数来实现与cmdlet类似的操作体验。
可能大家觉得编写函数还不简单么,随手就能写一个出来:
Function Hello($userName)
{
Write-Host
}
Hello Anna
即使是功能复杂的函数也大同小异,没什么复杂的。但实际上我们如果去仔细阅读帮助的话,会发现完整的函数是有以下形式构成的:
[] [([type]$parameter1[,[type]$parameter2])]
{
([type]$parameter1 [,[type]$parameter2])
dynamicparam {}
{}
{}
{}
}
下
面让我们来仔细看看上面的结构。函数名没什么好说的,但在函数名之前的要说一下,同变量一样,函数也是有作用域的,比较常用
的作用域有Global和Script。Global作用于整个PowerShell会话,只要PowerShell会话不结束,被Global修饰的变
量和函数都是可用的。而Script仅作用于脚本执行期间,一旦脚本执行完毕,脚本中被Script修饰的变量和函数都不在可用。以上内容,我们可以用下
面的代码进行演示:
$Script:userName1 =
$Global:userName2 =
Function Script:Script_Hello($userName)
{
Write-Host
}
Function Global:Global_Hello($userName)
{
Write-Host
}
Script_Hello $userName1
Script_Hello $userName2
演示效果如下:
接下来我们需要注意的是函数参数的放置位置,我们既可以使用和VBScript中函数类似的参数放置位置,也可以将参数放置在代码块中,只不过此时需要使用param。接着还需注意的是在PowerShell中还可以指定参数类型,我们接着用刚才的代码进行演示:
Function Script:Add1([int]$num1,[int]$num2)
{
$total = $num1 + $num2
Write-Host $total
}
Function Script:Add2
{
([int]$num1,[int]$num2)
$total = $num1 + $num2
Write-Host $total
}
add1 1 2
add2 1 2
add1 $number1 a
演示效果如下:
这里需要注意的是参数类型,如果PowerShell没有办法隐式的将参数值转换成相应的参数类型的话就会发生上面的错误。示例中的字符串显然没办法转换成整数。
接下来我们会看到dynamicparam {},dynamicparam 是Windows PowerShell v2中新增的关键字,在关于高级函数的最后一次介绍中,我们再来介绍它。下面我们继续往下看。
接
下来我们看到的是Begin、Process和End三个关键字。我们知道函数是可以接收来自管道的输入数据的,而我们可以通过Begin、
Process和End来控制函数如何去处理来自管道的输入数据。首先,Begin代表的代码块在函数开始执行时执行一次,并且在整个函数执行过程中只执
行一次,此时的函数还没有接收到来自管道的输入数据。接下来是Process关键字。Process代表的代码块将针对每一个输入对象运行一次,当
Process代码块正在处理数据时,每个对象都被赋值给$_变量。当函数接收完来自管道中的数据对象后,End代表的代码块将执行一次。如果我们没有在
函数中指定Begin、Process和End关键字,那么函数中的所有语句将被PowerShell默认为End代码块中的语句。
这
里我们还不得不提一下两个变量$_和$input。$_代表的是当前管道中的对象,而$input则代表来自管道的所有对象,并且这个变量仅在函数和脚本
块中使用。这两个变量将影响到我们在函数中如何处理数据。说了那么多,大家可能觉得头很晕,下面还是让我们来结合代码来看下这部分的介绍。具体代码如下:
Function Test-Function
{
{
Write-Host
}
{
Write-Host
}
{
Write-Host
}
}
1,2,3 | Test-Function
这部分代码很简单,我们将通过管道向函数传递三个数字,然后通过Write-Host来输出处理结果,通过输出的内容来考察Begin、Process和End是如何处理数据的。下面是示例代码的运行效果:
大
家首先看到在Begin代码执行后,PowerShell仅输出了提示信息“[Begin]:”,这是正常现象,因为此时函数还没有接收到来自管道的输入
数据,因此$input变量是空的。接下来我们看到输出了三行数据,这说明Process中的代码块对每一个来自管道的对象进行了处理。接下我们会发现最
后End代码块中的Write-Host没有输出$input变量的值,这是因此如果在我们的函数中存在Process代表的代码块的话,$input在
Process处理完之后将被清空。如果我们将示例代码中的Process代码块注释掉,然后再执行示例代码的话,大家会看到以下结果:
|
|
|