这应该是一个简单的任务,但是我已经看到过几次关于如何获取已执行cmdlet所在目录的路径的尝试,结果好坏参半。例如,当我执行C:\temp\myscripts\mycmdlet。我想能够将C:\temp\myscripts存储在mycmdlet.ps1中的一个变量中。

这是一个有效的解决方案(尽管有点麻烦):

$invocation = (Get-Variable MyInvocation).Value
$directorypath = Split-Path $invocation.MyCommand.Path
$settingspath = $directorypath + '\settings.xml'

另一个人建议这样的解决方案,它只适用于我们的测试环境:

$settingspath = '.\settings.xml'

我非常喜欢后一种方法,与每次都将文件路径解析为参数相比,我更喜欢这种方法,但我无法让它在我的开发环境中工作。我该怎么办?这与PowerShell的配置方式有关吗?


当前回答

试试这个:

$WorkingDir = Convert-Path .

其他回答

你还可以使用:

(Resolve-Path .\).Path

括号中的部分返回一个PathInfo对象。

(PowerShell 2.0起可用。)

在以下ide中调试时,大多数答案都不能工作:

PS-ISE (PowerShell ISE) VS Code (Visual Studio Code)

因为在这些$PSScriptRoot是空的,Resolve-Path .\(和类似的)将导致不正确的路径。

Freakydinde的答案是解决这些情况的唯一答案,所以我对其进行了投票,但我不认为该答案中的Set-Location是真正需要的。所以我修正了这个问题,让代码更清晰一些:

$directorypath = if ($PSScriptRoot) { $PSScriptRoot } `
    elseif ($psise) { split-path $psise.CurrentFile.FullPath } `
    elseif ($psEditor) { split-path $psEditor.GetEditorContext().CurrentFile.Path }

在Powershell 3及以上,您可以简单地使用

PSScriptRoot美元

这是我想到的。它是一个包含多个方法的数组,用于查找路径,使用当前位置,过滤空\空结果,并返回第一个非空值。

@((
  ($MyInvocation.MyCommand.Module.ModuleBase),
  ($PSScriptRoot),
  (Split-Path -Parent -Path $MyInvocation.MyCommand.Definition -ErrorAction SilentlyContinue),
  (Get-Location | Select-Object -ExpandProperty Path)
) | Where-Object { $_ })[0]

我喜欢单行解决方案:)

$scriptDir = Split-Path -Path $MyInvocation.MyCommand.Definition -Parent