Як запитати PowerShell, де щось є?
Наприклад, "який блокнот", і він повертає каталог, з якого запускається notepad.exe, відповідно до поточних шляхів.
Як запитати PowerShell, де щось є?
Наприклад, "який блокнот", і він повертає каталог, з якого запускається notepad.exe, відповідно до поточних шляхів.
Відповіді:
Першим псевдонімом, який я зробив, як тільки я почав налаштовувати свій профіль у PowerShell, був "який".
New-Alias which get-command
Щоб додати це до свого профілю, введіть це:
"`nNew-Alias which get-command" | add-content $profile
`N на початку останнього рядка - це гарантувати, що він розпочнеться як новий рядок.
Get-Command <command> | Format-Table Path, Name
тому я можу отримати шлях, де також сидить команда.
select -expandproperty Path
.
(gcm <command>).definition
для отримання шляху. gcm
псевдонім за замовчуванням для Get-Command
. Ви можете також використовувати групові символи, наприклад: (gcm win*.exe).definition
.
Ось фактичний * 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
okta
який вказує на сценарій Powershell, названий okta.ps1
не на моєму $PATH
. Використання прийнятої відповіді повертає ім'я сценарію ( okta -> okta.ps1
). Це нормально, але це не вказує мені місцезнаходження okta.ps1
. Однак, використовуючи цю відповідь, я отримую весь шлях ( C:\Users\blah\etc\scripts\okta.ps1
). Тож +1 від мене.
Я зазвичай просто набираю:
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
Ви отримуєте каталог і команду, яка відповідає тому, що ви шукаєте.
gcm note* | select CommandType, Name, Definition
. Якщо ви запускаєте його часто, ви, мабуть, мусите ввімкнути його у функції.
Спробуйте цей приклад:
(Get-Command notepad.exe).Path
(gcm py.exe).path
Моя пропозиція щодо функції, яка:
function which($cmd) { get-command $cmd | % { $_.Path } }
PS C:\> which devcon
C:\local\code\bin\devcon.exe
Швидкий і брудний матч 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.exe
і c:\windows\system32\notepad.exe
. Тому це особливо не підходить для форми $(which command)
. (Ще одна проблема полягає в тому, що воно буде надрукувати приємне, корисне повідомлення про помилку, якщо команда не буде знайдена, що також не буде добре розширюватися в $()
- це можна виправити /Q
, але не як псевдонім.)
where
здається , виконується пошук системної змінної PATH, а не поточної змінної PATH оболонки. Дивіться це запитання
Здається, це все, що ви хочете (я знайшов це на 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
Перевірте цю PowerShell Which .
Код, наданий там, говорить про це:
($Env:Path).Split(";") | Get-ChildItem -filter notepad.exe
Спробуйте where
команду в Windows 2003 або новішої версії (або Windows 2000 / XP, якщо ви встановили комплект ресурсів).
До речі, ця відповідь отримала більше відповідей на інші запитання:
where
псевдонім для Where-Object
командлету в Powershell, тому введення where <item>
підказки в Powershell нічого не дає. Таким чином, ця відповідь є абсолютно невірною - як зазначається у прийнятій відповіді в першому пов'язаному запитанні, щоб отримати DOS where
, потрібно набрати where.exe <item>
.
У 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 }
}
}
Використання:
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"
}
Або ця версія, викликаючи оригінал, де команда.
Ця версія також працює краще, оскільки вона не обмежується файлами 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'
}
Якщо ви хочете команду, яка приймає вхід з трубопроводу або як параметр, слід спробувати це:
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