ChatGPT解决这个技术问题 Extra ChatGPT

等效于 PowerShell 中的 *Nix 'which' 命令?

我如何询问 PowerShell 某物在哪里?

例如,“哪个记事本”,它会根据当前路径返回运行 notepad.exe 的目录。


P
Peter Mortensen

当我开始在 PowerShell 中自定义我的个人资料时,我创建的第一个别名是“which”。

New-Alias which get-command

要将其添加到您的个人资料中,请键入:

"`nNew-Alias which get-command" | add-content $profile

最后一行开头的 `n 是为了确保它将作为新行开始。


你可以把它放在你的配置文件脚本中。有关个人资料的更多信息 - msdn.microsoft.com/en-us/library/bb613488(VS.85).aspx
我喜欢运行:Get-Command <command> | Format-Table Path, Name,所以我也可以获得命令所在的路径。
有没有办法一直拥有路径而无需键入'|格式表路径,名称' ?
如果您想要 Unix 风格的行为来为您提供路径,您需要将 get-command 的输出通过管道传输到 select -expandproperty Path
仅使用 (gcm <command>).definition 获取路径。 gcmGet-Command 的默认别名。您还可以使用通配符,例如:(gcm win*.exe).definition
p
petrsnd

这是一个实际的*nix 等价物,即它提供*nix 风格的输出。

Get-Command <your command> | Select-Object -ExpandProperty Definition

只需替换您要查找的任何内容即可。

PS C:\> Get-Command notepad.exe | Select-Object -ExpandProperty Definition
C:\Windows\system32\notepad.exe

当您将其添加到您的配置文件时,您将希望使用函数而不是别名,因为您不能将别名与管道一起使用:

function which($name)
{
    Get-Command $name | Select-Object -ExpandProperty Definition
}

现在,当您重新加载您的个人资料时,您可以这样做:

PS C:\> which notepad
C:\Windows\system32\notepad.exe

我使用这种替代语法:“(Get-Command notepad).definition”
@B00merang 您的语法很棒-绝对更简洁-但不幸的是,即使删除了管道,也无法将其添加为别名,除非您包含要查找的程序的名称。
这是一篇旧文章,但如果有人被谷歌发送到这里(就像我一样),这个答案适用于比接受的答案更多类型的 Powershell 命令。例如,我有一个名为 okta 的别名,它指向不在我的 $PATH 上的名为 okta.ps1 的 Powershell 脚本。使用接受的答案返回脚本名称 (okta -> okta.ps1)。这没关系,但它没有告诉我 okta.ps1 的位置。然而,使用这个答案给了我整个路径(C:\Users\blah\etc\scripts\okta.ps1)。所以我+1。
D
David Mohundro

我通常只输入:

gcm notepad

或者

gcm note*

gcm 是 Get-Command 的默认别名。

在我的系统上,gcm note* 输出:

[27] » gcm note*

CommandType     Name                                                     Definition
-----------     ----                                                     ----------
Application     notepad.exe                                              C:\WINDOWS\notepad.exe
Application     notepad.exe                                              C:\WINDOWS\system32\notepad.exe
Application     Notepad2.exe                                             C:\Utils\Notepad2.exe
Application     Notepad2.ini                                             C:\Utils\Notepad2.ini

您将获得与您要查找的内容相匹配的目录和命令。


它有点乱,但比自定义函数和任意拆分更干净
当我在我的 powershell 命令提示符中键入“gcm notepad”时,我只得到前两列,第三列名为“ModuleName”,它是空的。您知道如何强制它默认列出“定义”列吗?
@PiyushSoni 这可能是因为 PowerShell 的更新版本。您始终可以通过执行 gcm note* | select CommandType, Name, Definition 之类的操作来显示其他列。但是,如果您经常运行它,您可能应该将它包装在一个函数中。
S
Shell

试试这个例子:

(Get-Command notepad.exe).Path

请添加更多代码或解释,以便 OP 更好地理解您。谢谢你。
感谢您添加更少的代码,所以我实际上可以记住一次:P
这就是我想要的!它也适用于 gcm:(gcm py.exe).path
P
Peter Mortensen

我对Which函数的提议:

function which($cmd) { get-command $cmd | % { $_.Path } }

PS C:\> which devcon

C:\local\code\bin\devcon.exe

这是一个比公认的更好的答案。它允许您添加上面建议的后处理后缀以提供更好的输出;别名没有。
C
Chris F Carroll

与 Unix which 的快速匹配是

New-Alias which where.exe

但是如果它们存在,它会返回多行,所以它变成

function which {where.exe command | select -first 1}

where.exe where 应该告诉你C:\Windows\System32\where.exe
where.exe 等同于 which -a,因为它将返回 所有 个匹配的可执行文件,而不仅仅是第一个要执行的可执行文件。也就是说,where.exe notepad 给出 c:\windows\notepad.exec:\windows\system32\notepad.exe。所以这特别适合$(which command)形式。 (另一个问题是,如果找不到该命令,它将打印一个很好的、有用的错误消息,它也不会在 $() 中很好地扩展——这可以用 /Q 来解决,但不能作为别名。)
采取的观点。我编辑了答案,但是是的,它不再是一个很好的解决方案
请注意,where 似乎在搜索系统 PATH 变量,而不是当前的 shell PATH 变量。请参阅this question
function which {where.exe $args[0] | select -first 1} 将使其可重复使用。此外,posh 在处理路径、引号和空格方面似乎更可靠,因此 function which {$(gcm $args[0]).source | select -first 1} 可能是更好的选择。
j
js2010

我喜欢 Get-Command | Format-List 或更短,对两者使用别名且仅用于 powershell.exe

gcm powershell | fl

你可以找到这样的别名:

alias -definition Format-List

制表符补全适用于 gcm

让选项卡一次列出所有选项:

set-psreadlineoption -editmode emacs

P
Peter Mortensen

这似乎可以满足您的要求(我在 http://huddledmasses.org/powershell-find-path/ 上找到了它):

Function Find-Path($Path, [switch]$All = $false, [Microsoft.PowerShell.Commands.TestPathType]$type = "Any")
## You could comment out the function stuff and use it as a script instead, with this line:
#param($Path, [switch]$All = $false, [Microsoft.PowerShell.Commands.TestPathType]$type = "Any")
   if($(Test-Path $Path -Type $type)) {
      return $path
   } else {
      [string[]]$paths = @($pwd);
      $paths += "$pwd;$env:path".split(";")

      $paths = Join-Path $paths $(Split-Path $Path -leaf) | ? { Test-Path $_ -Type $type }
      if($paths.Length -gt 0) {
         if($All) {
            return $paths;
         } else {
            return $paths[0]
         }
      }
   }
   throw "Couldn't find a matching path of type $type"
}
Set-Alias find Find-Path

但它并不是真正的“哪个”,因为它适用于任何文件(类型)并且找不到 cmdlet、函数或别名
P
Peter Mortensen

检查此PowerShell Which

那里提供的代码表明了这一点:

($Env:Path).Split(";") | Get-ChildItem -filter notepad.exe

我知道已经过去了好几年,但我的路径有“%systemroot%\system32\...”,而 PowerShell 不会扩展该环境变量并在执行此操作时引发错误。
P
Peter Mortensen

在 Windows 2003 或更高版本(或 Windows 2000/XP,如果您已安装资源工具包)上尝试 where 命令。

顺便说一句,这在其他问题中得到了更多答案:

Is there an equivalent of 'which' on Windows?

PowerShell equivalent to Unix which command?


where Powershell 中 Where-Object 命令行开关的别名,因此在 Powershell 提示符下键入 where <item> 不会产生任何结果。因此,这个答案是完全不正确的 - 如第一个链接问题中接受的答案所述,要获得 DOS where,您需要输入 where.exe <item>
A
Amin

如果您想要一个既可以接受来自管道的输入也可以作为参数的命令,您应该尝试以下操作:

function which($name) {
    if ($name) { $input = $name }
    Get-Command $input | Select-Object -ExpandProperty Path
}

将命令复制粘贴到您的配置文件 (notepad $profile)。

例子:

❯ echo clang.exe | which
C:\Program Files\LLVM\bin\clang.exe

❯ which clang.exe
C:\Program Files\LLVM\bin\clang.exe

P
Peter Mortensen

利用:

function Which([string] $cmd) {
  $path = (($Env:Path).Split(";") | Select -uniq | Where { $_.Length } | Where { Test-Path $_ } | Get-ChildItem -filter $cmd).FullName
  if ($path) { $path.ToString() }
}

# Check if Chocolatey is installed
if (Which('cinst.bat')) {
  Write-Host "yes"
} else {
  Write-Host "no"
}

还是这个版本,调用原来的where命令。

这个版本也更好用,因为它不限于 bat 文件:

function which([string] $cmd) {
  $where = iex $(Join-Path $env:SystemRoot "System32\where.exe $cmd 2>&1")
  $first = $($where -split '[\r\n]')
  if ($first.getType().BaseType.Name -eq 'Array') {
    $first = $first[0]
  }
  if (Test-Path $first) {
    $first
  }
}

# Check if Curl is installed
if (which('curl')) {
  echo 'yes'
} else {
  echo 'no'
}

G
George Ogden

您可以从 https://goprogram.co.uk/software/commands 安装 which 命令以及所有其他 UNIX 命令。


J
Jeff Zeitlin

我的 PowerShell 配置文件中有这个 which 高级功能:

    function which {
    <#
    .SYNOPSIS
    Identifies the source of a PowerShell command.
    .DESCRIPTION
    Identifies the source of a PowerShell command. External commands (Applications) are identified by the path to the executable
    (which must be in the system PATH); cmdlets and functions are identified as such and the name of the module they are defined in
    provided; aliases are expanded and the source of the alias definition is returned.
    .INPUTS
    No inputs; you cannot pipe data to this function.
    .OUTPUTS
    .PARAMETER Name
    The name of the command to be identified.
    .EXAMPLE
    PS C:\Users\Smith\Documents> which Get-Command
    
    Get-Command: Cmdlet in module Microsoft.PowerShell.Core
    
    (Identifies type and source of command)
    .EXAMPLE
    PS C:\Users\Smith\Documents> which notepad
    
    C:\WINDOWS\SYSTEM32\notepad.exe
    
    (Indicates the full path of the executable)
    #>
        param(
        [String]$name
        )
    
        $cmd = Get-Command $name
        $redirect = $null
        switch ($cmd.CommandType) {
            "Alias"          { "{0}: Alias for ({1})" -f $cmd.Name, (. { which $cmd.Definition } ) }
            "Application"    { $cmd.Source }
            "Cmdlet"         { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
            "Function"       { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
            "Workflow"       { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
            "ExternalScript" { $cmd.Source }
            default          { $cmd }
        }
    }

b
blenderfreaky

如果您有 scoop,则可以安装以下直接克隆:

scoop install which
which notepad

r
rayiik

也总是有使用 which 的选项。实际上有三种从 Windows powershell 访问 which 的方法,第一个(不一定是最好的)wsl -e which 命令(这需要为 Linux 安装 windows 子系统和正在运行的发行版)。 B. gnuwin32 是几个 .exe 格式的 gnu 二进制文件的端口,作为独立的捆绑启动器选项三,如果你去 /usr/bin 中安装它的位置安装 msys2(交叉编译器平台),你会发现很多许多更新的 gnu 实用程序。它们中的大多数都作为独立的 exe 工作,并且可以从 bin 文件夹复制到您的主驱动器的某个位置并添加到您的 PATH 中。


r
rayiik

也总是有使用 which 的选项。实际上有三种从 Windows powershell 访问的方法

第一个(虽然不是最好的)是 wsl(用于 linux 的 windows 子系统)

wsl -e which command 

这需要安装适用于 Linux 的 windows 子系统和正在运行的发行版。

接下来是 gnuwin32,它是多个 .exe 格式的 gnu 二进制文件的端口,作为独立的捆绑启动器

第三,安装 msys2(交叉编译器平台),如果你去 /usr/bin 中安装它的地方,你会发现许多更新的 gnu 实用程序。它们中的大多数都作为独立的 exe 工作,并且可以从 bin 文件夹复制到您的主驱动器的某个位置并添加到您的 PATH 中。