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


405

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

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

Ответы:


392

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

New-Alias which get-command

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

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

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


1
Вы можете поместить это в свой профиль скрипта. Подробнее о профилях - msdn.microsoft.com/en-us/library/bb613488(VS.85).aspx
Стивен Муравски

62
мне нравится бегать, Get-Command <command> | Format-Table Path, Nameпоэтому я могу получить путь, где находится команда тоже.
Jrsconfitto

4
Есть ли способ постоянно указывать путь без ввода '| Путь к формату таблицы, Имя '?
Гийом

10
Если вы хотите, чтобы поведение в стиле Unix указывало путь, вам нужно передать вывод команды get-command select -expandproperty Path.
Кейси

5
Используйте только (gcm <command>).definitionдля получения пути. gcmявляется псевдонимом по умолчанию для Get-Command. Вы можете также использовать групповые символы, например: (gcm win*.exe).definition.
Сачин Иосиф

165

Вот фактический * 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

22
Я использую этот альтернативный синтаксис: «(Get-Command notepad) .definition»
Янн

2
@ B00merang Ваш синтаксис великолепен - определенно более лаконичен - но, к сожалению, даже при удаленном канале он не может быть добавлен в качестве псевдонима, если вы не укажете название программы, которую вы ищете.
petrsnd

4
Это старый пост, но в случае, если кто-то отправит сюда Google (как я), этот ответ работает с большим количеством типов команд Powershell, чем принятый ответ. Например, у меня есть псевдоним, oktaкоторый указывает на скрипт Powershell okta.ps1, которого нет у меня $PATH. Использование принятого ответа возвращает имя скрипта ( okta -> okta.ps1). Это нормально, но это не говорит мне местоположение okta.ps1. Использование этого ответа, однако, дает мне весь путь ( C:\Users\blah\etc\scripts\okta.ps1). Так что +1 от меня.
Скай --- капитан

88

Я обычно просто набираю:

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

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


это немного грязно, но намного чище, чем пользовательские функции и произвольные разбиения
DevelopingChris

1
Когда я набираю «gcm notepad» в командной строке powershell, я просто получаю первые два столбца и третий столбец с именем «ModuleName», который пуст. Знаете ли вы, как заставить его по умолчанию перечислять столбец «Определение»?
Пиюш Сони

3
@PiyushSoni, вероятно, из-за обновленной версии PowerShell. Вы всегда можете отобразить другие столбцы, выполнив что-то вроде gcm note* | select CommandType, Name, Definition. Если вы запускаете его часто, вам, вероятно, следует обернуть его в функцию.
Дэвид Мохундро

40

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

(Get-Command notepad.exe).Path

2
Пожалуйста, добавьте больше кода или объяснения, чтобы ОП мог лучше вас понять. Спасибо.
sshashank124

3
Спасибо за добавление меньшего количества кода, так что я действительно могу вспомнить это на этот раз: P
albertjan

1
Это то, что я хотел! Он также работает с gcm:(gcm py.exe).path
Билл Эйджи

7

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

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

PS C:\> which devcon

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

Это лучший ответ, чем принятый. Это позволяет вам добавлять суффиксы постобработки, предложенные выше, чтобы обеспечить лучший вывод; псевдоним нет.
BobHy

5

Быстрый и грязный матч Unix whichявляется

New-Alias which where.exe

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

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

1
where.exe whereдолжен сказать вамC:\Windows\System32\where.exe
Крис Ф. Кэрролл

1
where.exeэквивалентно which -a, так как он вернет все соответствующие исполняемые файлы, а не только первый, который будет выполнен. То есть where.exe notepadдает c:\windows\notepad.exeи c:\windows\system32\notepad.exe. Так что это особенно не подходит для формы $(which command). (Другая проблема заключается в том, что в случае, если команда не найдена, она выведет полезное полезное сообщение об ошибке, которое также не будет расширяться $()- это можно исправить /Q, но не как псевдоним.)
Jeroen Mostert

Дело принято. Я отредактировал ответ, но да, это уже не так изящное решение
Крис Ф. Кэрролл,

1
Обратите внимание, что whereкажется, что поиск системной переменной PATH, а не текущей переменной PATH оболочки. Смотрите этот вопрос
Леонардо

3

Кажется, это делает то, что вы хотите (я нашел это на 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

Но на самом деле это не «что», так как он работает с любым файлом (типом) и не находит командлеты, функции или псевдонимы
Jaykul

3

Проверьте этот PowerShell, который .

Код, представленный там, предлагает это:

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

2
Я знаю, что прошло много лет, но мой путь был "% systemroot% \ system32 \ ...", и PowerShell не раскрывает эту переменную среды и выдает ошибки при этом.
TessellatingHeckler

3

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

gcm powershell | fl

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

alias -definition Format-List

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


2

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

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

Есть ли эквивалент «который» в Windows?

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


4
whereпсевдонимы Where-Objectкомандлета в Powershell, поэтому ввод where <item>в приглашении Powershell ничего не даст. Таким образом, этот ответ совершенно неверен - как отмечено в принятом ответе в первом связанном вопросе, чтобы получить DOS where, вам нужно набрать where.exe <item>.
Ян Кемп

0

У меня есть эта whichрасширенная функция в моем профиле PowerShell:

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 }
    }
}

0

Использование:

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'
}

0

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

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
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.