我需要安排一个PowerShell任务,如下所示:
powershell.exe -file ..\Execute\execute.ps1
我尝试将其分配给参数$Argument
,然后将其传递给schtasks
,如下所示:
$Argument = "powershell.exe -file '..\Execute\execute.ps1'"
schtasks /create /tn "SOE_Checks" /tr $Argument /sc DAILY /st 05:00 /ru "System" /rl HIGHEST /f
但是在运行上述代码后,什么也没发生-成功创建任务后,它似乎没有运行。
我也曾尝试将其分配给$Argument
,但不带引号,但仍然有效,但出现以下警告:
ERROR: Invalid syntax. Value expected for '/tr'.
Type "SCHTASKS /CREATE /?" for usage.
有人可以让我知道我在这里做错了什么吗? (我知道我可以使用PowerShell的New-ScheduledTaskAction
完成此操作,但我希望它能以这种方式工作)
只需添加一下,如果我将文件路径更改为$Argument
中的特定位置,如
$Argument = "powershell.exe -file 'C:\SOE\Execute\execute.ps1'"
,它可以正常工作而不会发出任何警告,但这并不理想。 / p>
我已经读过this,但对我不起作用
答案 0 :(得分:2)
使用schtasks.exe
创建的计划任务在工作目录设置为$env:windir\system32
[1] 的情况下执行,因此除非您的脚本恰好位于相对于那里的..\Execute\execute.ps1
,您的命令将无法正常工作。
如果您不想将脚本路径直接硬编码到命令中,请通过解析 absolute 动态构造命令 。 >当您分配给$Argument
时:
$Argument = 'powershell.exe -file \"{0}\"' -f (Convert-Path ..\Execute\execute.ps1)
请注意-不幸的是-需要将嵌入式"
转换为\"
,这是一个长期存在的错误,为了向后兼容,尚未解决-请参见{{3} }作为背景。
Convert-Path
将相对路径解析为绝对路径。
请注意,相对路径必须引用现有文件(或目录)。
类似地,相对路径在您的脚本也将相对于$env:windir\system32
;要使其相对于 script 的目录,请通过在开始时执行 Set-Location $PSScriptRoot
,首先更改为脚本的目录脚本。
注意:实际上,相同的规则适用于从Windows“运行”对话框运行命令(按 WinKey + R )时,< strong>您可以使用 test-drive 命令(该命令传递给schtasks /tr
,而不用外部引号,而不是整个schtasks
命令行)-但是请注意,工作目录将是用户的主目录,并且您将无法在PowerShell CLI的'...'
参数周围使用-File
-引号-参见下文):
cmd.exe
,这意味着:
例如,您不必担心cmd.exe
元字符的非双引号使用,例如&
,因此即使在单行中也可以使用这些字符引用的字符串作为powershell.exe
参数(的一部分)传递到PowerShell CLI -Command
。
相反,不直接支持输出重定向(例如> c:\path\to\log.txt
)。
在调用PowerShell CLI的上下文中,这意味着:
使用-File
,您不能在命令行上使用它们,而必须在脚本的 内执行。
使用-Command
,您可以使用它们,因为 PowerShell 才应用它们(但请注意,Windows PowerShell的{{ 1}}运算符会创建UTF-16LE文件。
(即使不涉及>
)对环境变量的引用的引用形式与{{1 }} 展开(例如cmd.exe
)
cmd.exe
不起作用-附加的%USERNAME%
被简单地视为一个文字,并且扩展仍然发生;例如%%
会产生%
。%%OS%%
(偶然)阻止扩展,但保留%Windows_NT%
-^%
不会 ;相反,它“破坏”了变量名,在这种情况下,令牌保持原样;例如,^
会产生^
,即按原样保留。以上内容适用于必须在计划任务中定义的命令,它们必须在任务计划程序(^%OS^%
)中交互查看或定义。
另外,用于从命令行/ PowerShell脚本/批处理文件创建计划任务:
您必须引用命令作为整体 和
符合调用外壳程序有关转义和预先字符串内插的语法规则。
(如果命令仅由不需要转义的单个单词组成,例如不包含空格或空格的可执行文件的路径,则只能引用 。特殊字符,并且不传递任何参数。)
调用^%OS^%
[2] 时,将taskschd.msc
参数作为一个整体引用如下:
PowerShell ,如果需要扩展(字符串插值)命令字符串 upfront ,请使用schtasks.exe
;否则,请使用/tr
。
重要:在两种情况下都需要将{em>嵌套 "..."
转换为'...'
,这在外部"
引用的情况下意味着嵌套的\"
必须转义为"..."
(原文如此)。
"
会识别嵌入 \`"
的引号并将其自动转换为schtasks.exe
的引号-这就是您原始命令的原因,即使在直接调用中,PowerShell CLI也不 支持将'...'
与"..."
结合使用,"powershell.exe -file '..\Execute\execute.ps1'"
仍然有效。 来自'...'
(无论是直接还是来自批处理文件),您必须使用-File
。
PowerShell示例:
以下PowerShell命令创建并执行两次
计划的任务,名为cmd.exe
和"..."
,在下一个日历分钟开始时在主叫用户的上下文中可见地运行。 (之后,您必须手动删除这些任务。)
您可能需要等待多达1分钟才能看到调用启动,这时将为每个任务弹出一个新的控制台窗口。
test1
[1]请注意,任务计划程序GUI允许您配置工作目录,但无法通过test2
实用程序使用此功能。
[2]传递给# Create sample script test.ps1 in the current dir. that
# echoes its arguments and then waits for a keypress.
'"Hi, $Args."; Read-Host "Press ENTER to exit"' > test.ps1
# Find the start of the next calendar minute.
$nextFullMinute = ([datetime]::Now.AddMinutes(1).TimeOfDay.ToString('hh\:mm'))
# -File example:
# Invoke test.ps1 and pass it 'foo' as an argument.
# Note the escaped embedded "..." quoting around the script path
# and that with -File you can only pass literal arguments at
# invocation time).
schtasks.exe /create /f /tn test1 /sc once /st $nextFullMinute `
/tr "powershell -File \`"$PWD/test.ps1\`" foo" #`# (dummy comment to fix broken syntax highlighting)
# -Command example:
# Invoke test.ps1 and pass it $env:USERNAME as an argument.
# Note the '...' around the script path and the need to invoke it with
# &, as well as the ` before $env:USERNAME to prevent its premature expansion.
schtasks.exe /create /f /tn test2 /sc once /st $nextFullMinute `
/tr "powershell -Command & '$PWD/test.ps1' `$env:USERNAME"
"Tasks will execute at ${nextFullMinute}:00"
PowerShell cmdlet的schtasks.exe
参数的值也是如此,尽管请注意,可执行文件名/路径是通过{{ 1}}参数。
相比之下,用于创建计划的PowerShell jobs 的-Argument
cmdlet接受 script块作为要运行的命令,从而消除了引人头痛的问题。 < / p>