Odpowiednik * Nix „które” polecenie w PowerShell?

404

Jak zapytać PowerShell, gdzie coś jest?

Na przykład „Który notatnik” i zwraca katalog, z którego uruchamiany jest notepad.exe zgodnie z bieżącymi ścieżkami.

DevelopisChris
źródło

Odpowiedzi:

390

Pierwszym pseudonimem, który utworzyłem, kiedy zacząłem dostosowywać swój profil w PowerShell, było „które”.

New-Alias which get-command

Aby dodać to do swojego profilu, wpisz:

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

`N na początku ostatniej linii oznacza, że ​​zacznie się jako nowa linia.

halr9000
źródło
1
Możesz umieścić go w skrypcie profilu. Więcej na temat profili - msdn.microsoft.com/en-us/library/bb613488(VS.85).aspx
Steven Murawski
61
Lubię biegać: Get-Command <command> | Format-Table Path, Namewięc mogę uzyskać ścieżkę, w której znajduje się również polecenie.
jrsconfitto
4
Czy jest jakiś sposób na ciągłą ścieżkę bez wpisywania „| Ścieżka do tabeli formatów, nazwa „?
Guillaume,
10
Jeśli chcesz, aby zachowanie w stylu uniksowym dawało ci ścieżkę, musisz przesłać dane wyjściowe polecenia get-command select -expandproperty Path.
Casey
5
Użyj, (gcm <command>).definitionaby uzyskać tylko ścieżki. gcmjest domyślnym aliasem dla Get-Command. Możesz także użyć symboli wieloznacznych, np (gcm win*.exe).definition. : .
Sachin Joseph
165

Oto rzeczywisty odpowiednik * nix, tzn. Daje wynik w stylu * nix.

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

Po prostu zamień na wszystko, czego szukasz.

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

Kiedy dodasz go do swojego profilu, będziesz chciał użyć funkcji zamiast aliasu, ponieważ nie możesz używać aliasów z potokami:

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

Teraz, gdy ponownie załadujesz swój profil, możesz to zrobić:

PS C:\> which notepad
C:\Windows\system32\notepad.exe
petrsnd
źródło
22
Używam tej alternatywnej składni: „(Notatnik Get-Command) .definicja”
Yann,
2
@ B00merang Twoja składnia jest świetna - zdecydowanie bardziej zwięzła - ale niestety, nawet po usunięciu potoku, nie można go dodać jako aliasu, chyba że podasz nazwę szukanego programu.
petrsnd
4
To jest stary post, ale na wypadek, gdyby ktoś został tu wysłany przez Google (tak jak ja), ta odpowiedź działa z większą liczbą poleceń Powershell niż zaakceptowana odpowiedź. Na przykład mam alias o nazwie, oktaktóry wskazuje na skrypt Powershell, okta.ps1którego nie ma na moim $PATH. Użycie zaakceptowanej odpowiedzi zwraca nazwę skryptu ( okta -> okta.ps1). To jest OK, ale nie mówi mi o lokalizacji okta.ps1. Użycie tej odpowiedzi daje mi jednak całą ścieżkę ( C:\Users\blah\etc\scripts\okta.ps1). Więc +1 ode mnie.
skye --- kapitan
87

Zwykle piszę:

gcm notepad

lub

gcm note*

gcm jest domyślnym aliasem Get-Command.

W moim systemie gcm note * wyświetla:

[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

Otrzymasz katalog i polecenie pasujące do tego, czego szukasz.

David Mohundro
źródło
jest nieco niechlujny, ale o wiele czystszy niż niestandardowe funkcje i arbitralne podziały
DevelopingChris
1
Kiedy wpisuję „gcm notepad” w wierszu polecenia programu PowerShell, otrzymuję tylko dwie pierwsze kolumny i trzecią kolumnę o nazwie „ModuleName”, która jest pusta. Czy wiesz, jak wymusić, aby domyślnie wyświetlała kolumnę „Definicja”?
Piyush Soni
3
@PiyushSoni, prawdopodobnie z powodu zaktualizowanej wersji programu PowerShell. Zawsze możesz wyświetlić inne kolumny, robiąc coś podobnego gcm note* | select CommandType, Name, Definition. Jeśli często go uruchamiasz, prawdopodobnie powinieneś zawinąć go w funkcję.
David Mohundro
40

Spróbuj tego przykładu:

(Get-Command notepad.exe).Path
thesqldev
źródło
2
Dodaj więcej kodu lub wyjaśnienia, aby PO mógł cię lepiej zrozumieć. Dziękuję Ci.
sshashank124
3
Dziękuję, że
dodaliście
1
Właśnie tego chciałem! Działa również z gcm:(gcm py.exe).path
Bill Agee
7

Moja propozycja dla funkcji Które:

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

PS C:\> which devcon

C:\local\code\bin\devcon.exe
VortiFred
źródło
To lepsza odpowiedź niż zaakceptowana. Pozwala dodać sugerowane powyżej sufiksy przetwarzania końcowego, aby zapewnić lepszą wydajność; alias nie.
BobHy
5

Szybkie i brudne dopasowanie do Uniksa whichto

New-Alias which where.exe

Ale zwraca wiele wierszy, jeśli one istnieją, więc się staje

function which {where.exe command | select -first 1}
Chris F. Carroll
źródło
1
where.exe wherepowinienem powiedziećC:\Windows\System32\where.exe
Chris F Carroll
1
where.exejest równoważne which -a, ponieważ zwróci wszystkie pasujące pliki wykonywalne, a nie tylko pierwszy do wykonania. To znaczy where.exe notepaddaje c:\windows\notepad.exei c:\windows\system32\notepad.exe. Więc jest to szczególnie nie nadaje się do postaci $(which command). (Innym problemem jest to, że wydrukuje miły, pomocny komunikat o błędzie, jeśli polecenie nie zostanie znalezione, które również nie będzie ładnie się rozwijało $()- co można naprawić /Q, ale nie jako alias.)
Jeroen Mostert
punkt wzięty. Zredagowałem odpowiedź, ale tak, to już nie jest tak fajne rozwiązanie
Chris F Carroll
1
Zauważ, że wherewydaje się , że szuka systemowej zmiennej PATH, a nie bieżącej zmiennej PATH powłoki. Zobacz to pytanie
Leonardo
3

Wydaje się, że robi to, co chcesz (znalazłem to na 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
Mikołaj
źródło
Ale tak naprawdę nie jest to „które”, ponieważ działa z dowolnym plikiem (typem) i nie znajduje poleceń cmdlet, funkcji ani aliasów
Jaykul
3

Sprawdź to PowerShell . Które .

Podany tam kod sugeruje to:

($Env:Path).Split(";") | Get-ChildItem -filter notepad.exe
tzot
źródło
2
Wiem, że minęły lata, ale moja ścieżka zawierała „% systemroot% \ system32 \ ...”, a PowerShell nie rozwija tej zmiennej środowiskowej i zgłasza błędy.
TessellatingHeckler
3

Lubię Get-Command | Format-Listlub krócej używam aliasów dla dwóch i tylko dla powershell.exe:

gcm powershell | fl

Możesz znaleźć takie aliasy:

alias -definition Format-List

Uzupełnianie tabulatorów działa z gcm.

js2010
źródło
2

Spróbuj where polecenie w systemie Windows 2003 lub nowszym (lub Windows 2000 / XP, jeśli zainstalowano zestaw zasobów).

BTW, otrzymałem więcej odpowiedzi na inne pytania:

Czy istnieje odpowiednik „który” w systemie Windows?

PowerShell równoważny z whichpoleceniem Unix ?

Anonimowy
źródło
4
wherealiasy do Where-Objectkomendy where <item>w Powershell, więc wpisanie monitu Powershell nic nie da. Ta odpowiedź jest więc całkowicie niepoprawna - jak zauważono w zaakceptowanej odpowiedzi w pierwszym łączonym pytaniu, aby uzyskać DOS where, musisz wpisać where.exe <item>.
Ian Kemp
0

Mam tę whichzaawansowaną funkcję w moim profilu 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 }
    }
}
Jeff Zeitlin
źródło
0

Posługiwać się:

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

Lub ta wersja, wywołując oryginalne polecenie where.

Ta wersja działa również lepiej, ponieważ nie ogranicza się do plików nietoperzy:

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'
}
Jerome
źródło
0

Jeśli chcesz comamnd, który akceptuje dane wejściowe z potoku lub jako parametr, powinieneś spróbować:

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

skopiuj i wklej polecenie do swojego profilu ( notepad $profile).

Przykłady:

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

 which clang.exe
C:\Program Files\LLVM\bin\clang.exe
Jestem w
źródło