Эквивалент команды * Nix 'which' в PowerShell?

Как мне спросить PowerShell, где что-то находится?

Например, «какой блокнот», и он возвращает каталог, из которого запускается notepad.exe, в соответствии с текущими путями.


person DevelopingChris    schedule 15.09.2008    source источник


Ответы (14)


Самым первым псевдонимом, который я сделал после того, как я начал настраивать свой профиль в PowerShell, был «which».

New-Alias which get-command

Чтобы добавить это в свой профиль, введите следующее:

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

`N в начале последней строки гарантирует, что она будет начинаться как новая строка.

person halr9000    schedule 15.09.2008
comment
Вы можете поместить это в свой скрипт профиля. Подробнее о профилях - msdn.microsoft.com/en-us /library/bb613488(VS.85).aspx - person Steven Murawski; 15.09.2008
comment
Мне нравится бегать: Get-Command <command> | Format-Table Path, Name, так что я тоже могу получить путь, по которому находится команда. - person jrsconfitto; 27.11.2012
comment
Есть ли способ иметь путь все время, не набирая '| Путь к форматной таблице, имя '? - person Guillaume; 11.01.2013
comment
В предыдущих версиях Get-Command не предоставлял отсортированный вывод. Теперь это так. Это явно правильный ответ сегодня ;-) - person Jaykul; 23.05.2014
comment
Если вы планируете использовать это в сценарии, вам следует передать некоторые дополнительные параметры, чтобы получить такое же поведение, как при прямом вызове в приглашении. Вот пример, если (! (Get-command node -ErrorAction SilentlyContinue)) {'узел не найден' | Write-Error} - person Sayed Ibrahim Hashimi; 25.10.2014
comment
Если вы хотите, чтобы поведение в стиле Unix давало вам путь, вам нужно направить вывод get-command в select -expandproperty Path. - person Casey; 29.07.2015
comment
Решение @Eld более верное по духу - person Jason Newton; 14.10.2015
comment
Как предлагает Джейсон, функция which(), определенная решение Eld существенно превосходит что-либо на предлагаю здесь. Если это не эквивалент Unix which, это неправильный ответ. - person Cecil Curry; 11.06.2016
comment
Используйте (gcm <command>).definition, чтобы получить только путь (пути). gcm - псевдоним по умолчанию для Get-Command. Вы также можете использовать подстановочные знаки, например: (gcm win*.exe).definition. - person Sachin Joseph; 28.02.2017
comment
В данный момент я изучаю ps, и мне кажется, что помимо того, что он может быть приличным языком сценариев, его довольно глупо использовать в командной строке. - person robert; 09.03.2018
comment
На всякий случай, если кому-то еще интересно, если вы заглянете в DLL автоматизации, Source вернет Definition, а Definition вернет Path, так что эти три свойства всегда будут эквивалентны (за исключением некоторых будущих изменений со стороны MS, конечно); - person Chad Schouggins; 11.04.2018
comment
для абсолютных новичков: чтобы это сработало, скопируйте и вставьте вторую команду в ответ в терминале PowerShell и запустите ее, а затем скопируйте и вставьте и запустите первую. - person M.Ionut; 16.07.2021

Вот фактический эквивалент * 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
person petrsnd    schedule 05.06.2013
comment
Я использую этот альтернативный синтаксис: (Блокнот Get-Command) .definition - person Yann; 19.12.2013
comment
@ B00merang У вас отличный синтаксис - определенно более краткий - но, к сожалению, даже после удаления канала его нельзя добавить в качестве псевдонима, если вы не укажете имя искомой программы. - person petrsnd; 26.02.2014
comment
Это старый пост, но в случае, если кто-то прислал сюда Google (как я), этот ответ работает с большим количеством типов команд Powershell, чем принятый ответ. Например, у меня есть псевдоним okta, который указывает на сценарий Powershell с именем okta.ps1, которого нет на моем $PATH. Использование принятого ответа возвращает имя сценария (okta -> okta.ps1). Это нормально, но он не сообщает мне местонахождение okta.ps1. Однако использование этого ответа дает мне весь путь (C:\Users\blah\etc\scripts\okta.ps1). Итак, +1 от меня. - person chris; 30.05.2019

Обычно я просто печатаю:

gcm notepad

or

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

Вы получаете каталог и команду, которая соответствует тому, что вы ищете.

person David Mohundro    schedule 15.09.2008
comment
это немного беспорядочно, но намного чище, чем пользовательские функции и произвольные разбиения - person DevelopingChris; 15.09.2008
comment
Когда я набираю gcm notepad в командной строке PowerShell, я получаю только первые два столбца и третий столбец с именем «ModuleName», который пуст. Вы знаете, как заставить его по умолчанию указывать столбец «Определение»? - person Piyush Soni; 28.11.2016
comment
@PiyushSoni, вероятно, из-за обновленной версии PowerShell. Вы всегда можете отобразить другие столбцы, выполнив что-то вроде gcm note* | select CommandType, Name, Definition. Однако, если вы запускаете его часто, вам, вероятно, следует обернуть его функцией. - person David Mohundro; 28.11.2016

Попробуйте этот пример:

(Get-Command notepad.exe).Path
person thesqldev    schedule 01.04.2014
comment
Пожалуйста, добавьте дополнительный код или объяснение, чтобы ОП мог лучше вас понять. Спасибо. - person sshashank124; 01.04.2014
comment
Спасибо, что добавили меньше кода, поэтому я могу вспомнить это хоть раз: P - person albertjan; 03.06.2015
comment
Это то, что я хотел! Он также работает с gcm: (gcm py.exe).path - person Bill Agee; 12.07.2016

Мое предложение для какой функции:

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

PS C:\> which devcon

C:\local\code\bin\devcon.exe
person VortiFred    schedule 17.11.2015
comment
Это лучший ответ, чем принятый. Это позволяет вам добавлять суффиксы постобработки, предложенные выше, чтобы обеспечить лучший результат; псевдоним - нет. - person BobHy; 12.12.2019

Быстрое совпадение с Unix which - это

New-Alias which where.exe

Но он возвращает несколько строк, если они существуют, поэтому он становится

function which {where.exe command | select -first 1}
person Chris F Carroll    schedule 08.04.2017
comment
where.exe where должен сказать вам C:\Windows\System32\where.exe - person Chris F Carroll; 08.04.2017
comment
where.exe эквивалентен which -a, так как он вернет все соответствующие исполняемые файлы, а не только первый, который будет выполнен. То есть where.exe notepad дает c:\windows\notepad.exe и c:\windows\system32\notepad.exe. Так что это особенно не подходит для формы $(which command). (Другая проблема заключается в том, что если команда не найдена, будет выведено красивое и полезное сообщение об ошибке, которое также не будет хорошо разворачиваться в $() - это можно исправить с помощью /Q, но не в качестве псевдонима.) - person Jeroen Mostert; 05.03.2018
comment
Дело принято. Я отредактировал ответ, но да, это уже не такое изящное решение - person Chris F Carroll; 07.03.2018
comment
Обратите внимание, что where, похоже, ищет системную переменную PATH, а не текущую переменную PATH оболочки. См. это вопрос - person Leonardo; 22.04.2020
comment
function which {where.exe $args[0] | select -first 1} сделает его многоразовым. Кроме того, posh кажется более надежным при работе с путями, кавычками и пробелами, поэтому function which {$(gcm $args[0]).source | select -first 1} может быть лучшим выбором. - person Nuno André; 23.02.2021

Мне нравится Get-Command | Format-List или короче, используя псевдонимы для двоих и только для powershell.exe:

gcm powershell | fl

Вы можете найти такие псевдонимы:

alias -definition Format-List

Завершение вкладок работает с gcm.

person js2010    schedule 11.04.2017

Кажется, это делает то, что вы хотите (я нашел его на 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
person Nicholas    schedule 15.09.2008
comment
Но это не совсем так, поскольку он работает с любым файлом (типом) и не находит командлеты, функции или псевдонимы. - person Jaykul; 23.05.2014

Проверьте этот PowerShell Which.

Приведенный там код предполагает следующее:

($Env:Path).Split(";") | Get-ChildItem -filter notepad.exe
person tzot    schedule 15.09.2008
comment
Я знаю, что прошли годы, но на моем пути был% systemroot% \ system32 \ ..., а PowerShell не расширяет эту переменную среды и при этом выдает ошибки. - person TessellatingHeckler; 27.03.2014

Попробуйте выполнить команду where в Windows 2003 или более поздней версии (или Windows 2000 / XP, если вы установили набор ресурсов).

Кстати, это получило больше ответов на другие вопросы:

Есть ли эквивалент «which» в Windows?

эквивалент PowerShell команде which Unix?

person Anonymous    schedule 13.12.2011
comment
where псевдонимы Where-Object командлета в Powershell, поэтому ввод where <item> в командной строке Powershell ничего не дает. Таким образом, этот ответ совершенно неверен - как указано в принятом ответе в первом связанном вопросе, чтобы получить DOS where, вам нужно ввести where.exe <item>. - person Ian Kemp; 01.07.2015

Использовать:

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'
}
person Jerome    schedule 22.12.2013

Если вам нужна команда, которая принимает входные данные из конвейера или как параметр, вы должны попробовать следующее:

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
person Amin    schedule 01.06.2020

Вы можете установить команду which из https://goprogram.co.uk/software/commands вместе со всеми другими командами UNIX.

person George Ogden    schedule 20.07.2020

В моем профиле 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 }
        }
    }
person Jeff Zeitlin    schedule 05.03.2018