Answers:
在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
,该别名指向okta.ps1
不在我的Powershell脚本上$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
与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
,但不能作为别名。)
这似乎可以满足您的要求(我在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
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"
}
或此版本,调用原始的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'
}
如果您想要一个既接受管道输入又作为参数输入的命令,则应尝试以下操作:
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