เทียบเท่า * Nix 'ซึ่งคำสั่ง' ใน PowerShell?


404

ฉันจะถาม PowerShell ได้ที่ไหน

ตัวอย่างเช่น "notepad" ซึ่งจะส่งคืนไดเรกทอรีที่ notepad.exe ถูกเรียกใช้จากเส้นทางปัจจุบัน

คำตอบ:


390

นามแฝงแรกที่ฉันทำเมื่อฉันเริ่มกำหนดโปรไฟล์ของฉันเองใน 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
Steven Murawski

61
ฉันชอบวิ่ง: Get-Command <command> | Format-Table Path, Nameดังนั้นฉันจะได้รับเส้นทางที่คำสั่งอยู่ด้วย
jrsconfitto

4
มีวิธีใดบ้างที่จะมีพา ธ ตลอดเวลาโดยไม่ต้องพิมพ์ '| รูปแบบตารางพา ธ ชื่อ '?
Guillaume

10
หากคุณต้องการพฤติกรรม Unix select -expandproperty Pathสไตล์ให้คุณเส้นทางที่คุณจะต้องท่อส่งออกของได้รับคำสั่งไปยัง
Casey

5
ใช้(gcm <command>).definitionเพื่อรับเส้นทางเท่านั้น เป็นนามแฝงเริ่มต้นสำหรับgcm นอกจากนี้คุณยังสามารถใช้สัญลักษณ์แทนเช่น:Get-Command (gcm win*.exe).definition
Sachin Joseph

165

นี่คือ * nix ที่เทียบเท่าได้จริงนั่นคือให้เอาต์พุต * nix-style

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

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 เป็นนามแฝงเริ่มต้นสำหรับรับคำสั่ง

ในระบบของฉันเอาต์พุต 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' ซึ่งว่างเปล่า คุณรู้วิธีบังคับให้รายการคอลัมน์ 'ข้อกำหนด' เป็นค่าเริ่มต้นหรือไม่
Piyush Soni

3
@PiyushSoni นั่นอาจเป็นเพราะ PowerShell รุ่นปรับปรุงแล้ว คุณสามารถแสดงคอลัมน์อื่น ๆ gcm note* | select CommandType, Name, Definitionด้วยการทำสิ่งที่ชอบ หากคุณใช้งานบ่อยคุณควรห่อไว้ในฟังก์ชั่น
David Mohundro

40

ลองตัวอย่างนี้:

(Get-Command notepad.exe).Path

2
โปรดเพิ่มรหัสหรือคำอธิบายเพิ่มเติมเพื่อให้ OP สามารถเข้าใจคุณดีขึ้น ขอบคุณ.
sshashank124

3
ขอบคุณที่เพิ่มโค้ดให้น้อยลงดังนั้นฉันจึงสามารถจดจำได้ในครั้งเดียว: P
albertjan

1
นี่คือสิ่งที่ฉันต้องการ! มันทำงานร่วมกับ gcm เช่นกัน:(gcm py.exe).path
Bill Agee

7

ข้อเสนอของฉันสำหรับฟังก์ชั่น Which:

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

PS C:\> which devcon

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

นี่เป็นคำตอบที่ดีกว่าคำตอบที่ได้รับการยอมรับ ช่วยให้คุณสามารถเพิ่มคำต่อท้าย postprocessing ที่แนะนำข้างต้นเพื่อให้ได้ผลลัพธ์ที่ดีขึ้น นามแฝงไม่ได้
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
Chris F Carroll

1
where.exeเทียบเท่ากับwhich -aซึ่งจะให้ผลตอบแทนที่ตรงกันทั้งหมดไม่ใช่แค่ตัวแรกที่ถูกเรียกใช้ นั่นคือการwhere.exe notepadให้และc:\windows\notepad.exe c:\windows\system32\notepad.exeดังนั้นนี่คือโดยเฉพาะอย่างยิ่งไม่$(which command)เหมาะสำหรับรูปแบบ (ปัญหาอื่นคือมันจะพิมพ์ข้อความแสดงข้อผิดพลาดที่ดีและมีประโยชน์หากไม่พบคำสั่งซึ่งจะไม่ขยายอย่างดีใน$()- ซึ่งสามารถแก้ไขได้/Qแต่ไม่ได้เป็นนามแฝง)
Jeroen Mostert

จุดที่ถ่าย ฉันแก้ไขคำตอบ แต่ใช่ก็ไม่แก้ปัญหาเรียบร้อยดังนั้น
คริสคาร์โรลล์ F

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

แต่ไม่ใช่ "ที่" เพราะทำงานกับไฟล์ใด ๆ (ชนิด) และไม่พบ cmdlets, ฟังก์ชั่นหรือชื่อแทน
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)

BTW นี้ได้รับคำตอบเพิ่มเติมในคำถามอื่น ๆ :

มี 'เทียบเท่า' บน Windows หรือไม่

PowerShell เทียบเท่ากับwhichคำสั่งUnix ?


4
whereนามแฝงกับWhere-Objectcommandlet ใน Powershell ดังนั้นการพิมพ์where <item>ใน Powershell prompt จะไม่ได้ผลอะไรเลย คำตอบนี้จึงไม่ถูกต้องสมบูรณ์ - ตามที่ระบุไว้ในคำตอบที่ได้รับการยอมรับในคำถามแรกที่เชื่อมโยงที่จะได้รับ DOS where, where.exe <item>คุณจำเป็นต้องพิมพ์
Ian Kemp

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

หากคุณต้องการ comamnd ที่ทั้งสองยอมรับอินพุตจากไปป์ไลน์หรือเป็นพารามิเตอร์คุณควรลอง:

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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.