PowerShell2.0之与COM对象交互(五)与脚本宿主代码协同工作


网络中存在大量由VBScript和Jscript或者其他Windows脚本宿主相关的语言编写的代码,其中相当一部分用对应语言实现很简短,而且有部分代码甚至不能转换为PowerShell代码。丢弃所有这些代码并且完全使用PowerShell重写很不明智,因此需要在PowerShell中有一种重用Jscript和VBScript脚本的代码方法。

MSScriptControl这个COM对象可以作为脚本环境的宿主,其ProgID是MSScriptControl.ScriptControl。用户可以用其注册并执行代码,使用这个对象作为从PowerShell中调用现有VBScript和Jscript脚本的方法。一旦导入脚本,即可调用其中的函数,传递参数并返回值。

为了示范脚本如何控制对象的操作,下面演示计算文件大小的两个实例,分别是Jscript编写的FileSize.js和VBScirpt编写的FileSize.vbs。两个脚本中包括定义为“GetFileSize()”的函数接受一个文件路径为参数,并返回文件路径及其大小。Jscript脚本的源代码如下:

function GetFileSize(filePath)

{

var fileSystem = new ActiveXObject("Scripting.FileSystemObject");

var file = fileSystem.GetFile(filePath);

return (filePath + " has " + file.Size + " bytes.");

}

VBScript脚本的代码如下:

Function GetFileSize(filePath)

Dim fileSystem

Set fileSystem = CreateObject("Scripting.FileSystemObject")

Dim file

Set file = fileSystem.GetFile(filePath)

GetFileSize = filePath & " has " & file.Size & " bytes."

End Function

能够看到每种语言都有其相对特殊的函数声明方法、变量,以及与COM对象操作的方法。为了能够从PowerShell中调用Jscript函数,需要读取FileSize.js文件内容并且传递给脚本空间的AddCode()方法。这个方法将会执行函数定义,并使其在后面可用。

1 执行代码

在将脚本代码包含在脚本控件中后,即可调用在Jscript中定义的函数,为此可以使用Eval()或者Run()方法。Eval()方法将字符串作为输入并执行,其返回值将会把结果返回给调用方,本实例为PowerShell;Run()方法可以接受多个参数,其中第1个是函数名,后续是将要传递给函数的参数,该方法返回函数的返回值给PowerShell的脚本。在这里创建一个名为“JScriptEval.ps1”的脚本,分别使用Eval()和Run()调用Jscript函数,代码如下:

$jscript = New-Object -COM MSScriptControl.ScriptControl

$jscript.Language = "JScript"

$jsLines = Get-Content "FileSize.js"

$jsCode = [string]::Join("`n", $jsLines)

$jscript.AddCode($jsCode)

$fileName = (dir FileSize.js).FullName

Write-Host "Using Eval"

$jscript.Eval("GetFileSize(`"$($fileName.Replace(‘\’, ‘\\’))`")")

Write-Host "Using Run"

$jscript.Run("GetFileSize", $fileName)

其中将Language属性设置为“JScript”,表示其后调用的所有代码都是Jscript。接下来用Get-Content cmdlet获取FileSize.js文件中包含的所有内容,然后为各行代码添加换行符,并将其拼接后传递给AddCode()方法。

Eval()方法需要将双引号转义为Jscript字符串,把反斜杠替换为双反斜杠。因为反斜杠在Jscript中是一个有特殊含义的转义字符,需要被转义;Run()方法不需要执行任何转义操作,因为使用的是脚本控件的机制传递参数。图1所示为该脚本的执行结果。

image

图1 执行结果

Eval()和Run()方法的执行结果相同,使用Run()方法可避免很多不必要的麻烦。

下面创建一个名为“VBScriptEval.ps1”的脚本,通过调用VBScript代码定义的GetFileSize()函数,代码如下:

$vbscript = New-Object -COM MSScriptControl.ScriptControl

$vbscript.Language = "VBScript"

$vbsLines = Get-Content "FileSize.vbs"

$vbsCode = [string]::Join("`n", $vbsLines)

$vbscript.AddCode($vbsCode)

$fileName = (dir FileSize.vbs).FullName

Write-Host "Using Eval"

$vbscript.Eval("GetFileSize(`"$fileName`")")

Write-Host "Using Run"

$vbscript.Run("GetFileSize", $fileName)

这个脚本与前面Jscript版本的最大不同是创建脚本控件对象的Language属性值。需要强调的一点是在VBScript脚本中调用Eval()方法时,不需要转义反斜杠字符,因其在VBScript中并不包含特殊含义。该脚本的执行结果如图2所示。

image

图2 执行结果

2 在MSScriptControl中公开对象

Eval()和Run()方法为其他语言提供了很好的访问途径,但其表现不像真正的方法。如果能调用脚本控件对象的特性,即动态对象生成,即可很好地解决这个问题。MSScriptControl对象有一个CodeObject属性,用于将附加到脚本控件的代码返回一个匿名对象,代码对象将会公开作为公开方法添加的所有函数,这样即可使得外部脚本对象的操作与常规的.NET对象毫无差别。

下面使用该属性调用JScript函数,创建一个新的脚本文件“JScriptCodeObject.ps1”。配置一个脚本对象,通过动态生成的对象调用其函数,代码如下:

$jscript = New-Object -COM MSScriptControl.ScriptControl

$jscript.Language = "JScript"

$jsLines = Get-Content "FileSize.js"

$jsCode = [string]::Join("`n", $jsLines)

$jscript.AddCode($jsCode)

$fileName = (dir FileSize.js).FullName

$fileSize = $jscript.CodeObject

$fileSize.GetFileSize($fileName)

脚本的执行结果如图3所示。

image

图3 执行结果

接下对比VBScript版本,下面是脚本VBScriptCodeObject.ps1的代码:

$vbscript = New-Object -COM MSScriptControl.ScriptControl

$vbscript.Language = "VBScript"

$vbsLines = Get-Content "FileSize.vbs"

$vbsCode = [string]::Join("`n", $vbsLines)

$vbscript.AddCode($vbsCode)

$fileName = (dir FileSize.vbs).FullName

$fileSize = $vbscript.CodeObject

$fileSize.GetFileSize($fileName)

执行结果如图4所示。

image

图4 执行结果

3 总 结

PowerShell为COM对象提供了近乎完美的支持,本文的实例涉及主要的交互性操作,如使用集合和索引属性。只要不同的应用程序和服务已经具有公开的COM自动化接口,则可尝试使用COM来提高工作效率。本文介绍了针对Windows脚本宿主的自动化编程,用户可以通过以上的介绍扩展到对其他COM对象进行操作。

 

作者: 付海军
出处:http://fuhj02.blog.51cto.com
版权:本文版权归作者和51cto共有
转载:欢迎转载,为了保存作者的创作热情,请按要求【转载】,谢谢
要求:未经作者同意,必须保留此段声明;必须在文章中给出原文连接;否则必究法律责任
个人网站: http://txj.shell.tor.hu/


发表回复