ChatGPT解决这个技术问题 Extra ChatGPT

如何将 PowerShell 与 Visual Studio 命令提示符一起使用?

我使用 Beta 2 已经有一段时间了,运行 Visual Studio 2010 命令提示符时我不得不使用 cmd.exe 让我发疯。我曾经为 Visual Studio 2008 提供了一个不错的 vsvars2008.ps1 脚本。是否有 vsvars2010.ps1 脚本或类似的脚本?

从 Visual Studio 2022 开始,已经有一个专门的快捷方式可以设置 PowerShell 进行开发。 Windows 11 上包含 VS 2022 的 Cmd 和 PowerShell 快捷方式的目录的默认路径是 C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Visual Studio 2022\Visual Studio Tools。注意:开箱即用的快捷方式会打开 Windows PowerShell(即 5.1 版),但可以轻松地将其更改为 pwsh.exe 以使用最新最好的 PowerShell 7

M
Matt Dillard

从博文中大量窃取Replace Visual Studio Command Prompt with PowerShell,我能够让它发挥作用。我将以下内容添加到我的 profile.ps1 文件中,一切顺利。

pushd 'c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC'
cmd /c "vcvarsall.bat&set" |
foreach {
  if ($_ -match "=") {
    $v = $_.split("="); set-item -force -path "ENV:\$($v[0])"  -value "$($v[1])"
  }
}
popd
write-host "`nVisual Studio 2010 Command Prompt variables set." -ForegroundColor Yellow

这多年来一直运作良好 - 直到 Visual Studio 2015。vcvarsall.bat 不再存在。相反,您可以使用位于 Common7\Tools 文件夹中的 vsvars32.bat 文件。

pushd 'C:\Program Files (x86)\Microsoft Visual Studio 14.0\Common7\Tools'    
cmd /c "vsvars32.bat&set" |
foreach {
  if ($_ -match "=") {
    $v = $_.split("="); set-item -force -path "ENV:\$($v[0])"  -value "$($v[1])"
  }
}
popd
write-host "`nVisual Studio 2015 Command Prompt variables set." -ForegroundColor Yellow

Visual Studio 2017 的情况再次发生变化。vsvars32.bat 似乎已被弃用,取而代之的是 VsDevCmd.bat。确切路径可能因您使用的 Visual Studio 2017 版本而异。

pushd "C:\Program Files (x86)\Microsoft Visual Studio\2017\Enterprise\Common7\Tools"
cmd /c "VsDevCmd.bat&set" |
foreach {
  if ($_ -match "=") {
    $v = $_.split("="); set-item -force -path "ENV:\$($v[0])"  -value "$($v[1])"
  }
}
popd
Write-Host "`nVisual Studio 2017 Command Prompt variables set." -ForegroundColor Yellow

您还可以使拆分仅创建两个项目以避免破坏值,包括等号,它也是环境变量名称和值的分隔符:

$v = $_.split("=", 2); set-item -force -path "ENV:\$($v[0])"  -value 

Visual Studio 2022 的微小更改,现在它是 64 位。

pushd "C:\Program Files\Microsoft Visual Studio\2022\Enterprise\Common7\Tools"
cmd /c "VsDevCmd.bat&set" |
foreach {
  if ($_ -match "=") {
    $v = $_.split("=", 2); set-item -force -path "ENV:\$($v[0])"  -value "$($v[1])" 
  }
}
popd
Write-Host "`nVisual Studio 2022 Command Prompt variables set." -ForegroundColor Yellow

请注意,同样的技术适用于 Visual Studio 2012。只需将“Microsoft Visual Studio 10.0”更改为“Microsoft Visual Studio 11.0”
echo $Profile 查看您的 profile.ps1 的预期路径(如果您从未创建过它)
脚本本身运行良好。不过需要注意的是(可能):由于 Visual Studio 中的包管理器控制台本身就是一个 PowerShell 主机,因此该脚本也将在那里运行。这似乎不是问题,除非您注意到“在没有调试的情况下运行”或任何其他运行启动标准 Windows 控制台的功能或插件在 PMC 初始化后将不起作用。我解决了这个问题,而不是将此答案中的脚本保存到“profile.ps1”,而是将其保存到“Microsoft.PowerShell_profile.ps1”,以便它只能在“正确的”PowerShell会话中运行。
当有完美的环境变量(VS2015 的 VS140COMNTOOLS)可供使用时,硬编码路径是非常糟糕的做法。这甚至适用于自定义 VS 安装。
我很欣赏使用环境变量的愿望,但这些变量似乎是由我们试图从中提取变量的批处理文件初始化的。我很乐意看到相反的证据。在执行 VsDevCmd.bat 之前,我有一个干净的 Windows 10 安装和一个干净的 Visual Studio 2017 安装,并且没有 VS150COMNTOOLS 环境变量。
P
Peter Mortensen

最简单的选择是运行 VS 2010 命令提示符,然后启动 PowerShell.exe。如果您真的想从“家庭”PowerShell 提示符执行此操作,那么您展示的方法就是要走的路。我使用了 Lee Holmes 不久前写的一个脚本:

<#
.SYNOPSIS
   Invokes the specified batch file and retains any environment variable changes
   it makes.
.DESCRIPTION
   Invoke the specified batch file (and parameters), but also propagate any
   environment variable changes back to the PowerShell environment that
   called it.
.PARAMETER Path
   Path to a .bat or .cmd file.
.PARAMETER Parameters
   Parameters to pass to the batch file.
.EXAMPLE
   C:\PS> Invoke-BatchFile "$env:VS90COMNTOOLS\..\..\vc\vcvarsall.bat"
   Invokes the vcvarsall.bat file to set up a 32-bit dev environment.  All
   environment variable changes it makes will be propagated to the current
   PowerShell session.
.EXAMPLE
   C:\PS> Invoke-BatchFile "$env:VS90COMNTOOLS\..\..\vc\vcvarsall.bat" amd64
   Invokes the vcvarsall.bat file to set up a 64-bit dev environment.  All
   environment variable changes it makes will be propagated to the current
   PowerShell session.
.NOTES
   Author: Lee Holmes
#>
function Invoke-BatchFile
{
   param([string]$Path, [string]$Parameters)

   $tempFile = [IO.Path]::GetTempFileName()

   ## Store the output of cmd.exe.  We also ask cmd.exe to output
   ## the environment table after the batch file completes
   cmd.exe /c " `"$Path`" $Parameters && set > `"$tempFile`" "

   ## Go through the environment variables in the temp file.
   ## For each of them, set the variable in our local environment.
   Get-Content $tempFile | Foreach-Object {
       if ($_ -match "^(.*?)=(.*)$")
       {
           Set-Content "env:\$($matches[1])" $matches[2]
       }
   }

   Remove-Item $tempFile
}

注意:此功能将在即将推出的 PowerShell Community Extensions 2.0 基于模块的版本中提供。


P
Peter Mortensen

我找到了一个简单的方法here:修改快捷方式。

原来的快捷方式是这样的:

%comspec% /k ""C:\Program Files (x86)\Microsoft Visual Studio 12.0\Common7\Tools\VsDevCmd.bat""

在最后一个引号之前添加 & powershell,如下所示:

%comspec% /k ""C:\Program Files (x86)\Microsoft Visual Studio 12.0\Common7\Tools\VsDevCmd.bat" & powershell"

如果您想让它看起来更像 PowerShell,请转到快捷方式属性的颜色选项卡,并将红色、绿色和蓝色值分别设置为 1、36 和 86。


直截了当,像魅力一样工作。这个答案值得更多的赞成。
我唯一不喜欢的是,它会加载一个冗余的 cmd.exe 进程。除此之外,这是一个不错的解决方案。
P
Peter Mortensen

(a) 提供 Visual Studio 2013 支持; (b) 结合前两个答案中最好的一个; (c) 提供一个函数包装器:

这建立在 Andy's technique 的基础上(它建立在 Allen Mack 的技术之上,正如 Andy 所指出的那样(这反过来又建立在 Robert Anderson 的技术之上,正如 Allen 所指出的那样(所有这些都有一个轻微的故障,正如仅称为“me- -”,所以我也考虑到了这一点)))。

这是我的最终代码——注意在正则表达式中使用非贪婪量词来处理值中任何可能的嵌入等于。这也恰好简化了代码:单个匹配而不是匹配然后拆分,如 Andy 的示例或匹配然后 indexof 和子字符串,如“me--”的示例)。

function Set-VsCmd
{
    param(
        [parameter(Mandatory, HelpMessage="Enter VS version as 2010, 2012, or 2013")]
        [ValidateSet(2010,2012,2013)]
        [int]$version
    )
    $VS_VERSION = @{ 2010 = "10.0"; 2012 = "11.0"; 2013 = "12.0" }
    $targetDir = "c:\Program Files (x86)\Microsoft Visual Studio $($VS_VERSION[$version])\VC"
    if (!(Test-Path (Join-Path $targetDir "vcvarsall.bat"))) {
        "Error: Visual Studio $version not installed"
        return
    }
    Push-Location $targetDir
    cmd /c "vcvarsall.bat&set" |
    foreach {
      if ($_ -match "(.*?)=(.*)") {
        Set-Item -force -path "ENV:\$($matches[1])" -value "$($matches[2])"
      }
    }
    Pop-Location
    Write-Host "`nVisual Studio $version Command Prompt variables set." -ForegroundColor Yellow
}

对于 PowerShell 2.0 兼容性,参数部分需要 [parameter(Mandatory=$true, ...
很好,但是如果没有 pushd/popd,imo 会更好。只需使用 cmd /c """$targetDir\vcvarsall.bat""&set" 之类的东西
P
Peter Mortensen

Keith has already mentioned PowerShell 社区扩展 (PSCX),及其 Invoke-BatchFile 命令:

Invoke-BatchFile "${env:ProgramFiles(x86)}\Microsoft Visual Studio 12.0\VC\vcvarsall.bat"

我还注意到 PSCX 也有一个 Import-VisualStudioVars 函数:

Import-VisualStudioVars -VisualStudioVersion 2013

从 PSCX 3.2.0 开始,此 cmdlet 不支持 VS 2015。我为它打开了一个 issue
P
Peter Mortensen

感谢Andy S for his answer。我一直在使用他的解决方案,但今天遇到了问题。任何包含等号的值都会在等号处被截断。例如,我有:

JAVA_TOOL_OPTIONS=-Duser.home=C:\Users\Me

但我的 PowerShell 会话报告:

PS C:\> $env:JAVA_TOOL_OPTIONS

-Duser.home

我通过将我的配置文件脚本修改为以下内容来解决此问题:

pushd 'c:\Program Files (x86)\Microsoft Visual Studio 11.0\VC'
cmd /c "vcvarsall.bat&set" |
foreach {
  if ($_ -match "=") {
    $i = $_.indexof("=")
    $k = $_.substring(0, $i)
    $v = $_.substring($i + 1)
    Set-Item -force -path "ENV:\$k"  -value "$v"
  }
}
popd

嘿,不错的补充。也用 VS2015/17 versions 更新它?这是第一个谷歌结果,我认为你的添加会对人们有所帮助。
A
AlQuemist

使用“模块”和 VsDevShell 更简单、更简洁。下面是一个适用于 MSVC 和 Clang 的简单 Powershell 脚本:

param(
    [String] $Compiler = "MSVC"
)

if ($Compiler -ne "MSVC" -and $Compiler -ne "Clang") {
    Write-Error "Unknown compiler '$Compiler'; must be MSVC or Clang"
    Exit -1
}

Write-Host "======================================="
Write-Host "Setting up environment variables..."

# Visual Studio path <https://github.com/microsoft/vswhere/wiki/Find-VC>
$vsPath = &"${env:ProgramFiles(x86)}\Microsoft Visual Studio\Installer\vswhere.exe" -latest -products * -requires Microsoft.VisualStudio.Component.VC.Tools.x86.x64 -property installationpath

Write-Host "Microsoft Visual Studio path = '$vsPath'"

# Use module `Microsoft.VisualStudio.DevShell.dll`
Import-Module (Get-ChildItem $vsPath -Recurse -File -Filter Microsoft.VisualStudio.DevShell.dll).FullName

Enter-VsDevShell -VsInstallPath $vsPath -SkipAutomaticLocation -DevCmdArguments '-arch=x64'

# NOTE: `-DevCmdArguments` are arguments to `vsdevcmd.bat`

# Select compiler
if ($Compiler -eq "MSVC") {
    $_Compiler = "MSVC"
    Set-Item -Path "env:CC" -Value "cl.exe"
    Set-Item -Path "env:CXX" -Value "cl.exe"
}
elseif ($Compiler -eq "Clang") {
    $_Compiler = "Clang"
    Set-Item -Path "env:CC" -Value "clang-cl.exe"
    Set-Item -Path "env:CXX" -Value "clang-cl.exe"
}

Write-Host "Selecting $_Compiler as C/C++ compiler."
Write-Host "======================================="

G
Glavo

事实上,我们有一个简单而干净的方法来做到这一点。

众所周知,在shell中启动的子shell会自动继承父shell的变量。因此,我们只需要在PowerShell中启动CMD,执行vcvars64.bat,在这个CMD中再次启动PowerShell即可。

对于 Visual Studio 2019:

cmd /K '"C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat" && powershell.exe'

如果您使用的是 PowerShell 核心:

cmd /K '"C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat" && pwsh.exe'

您可以将其包装为一个函数并将其放入 Powershell Profiles(所有用户为 $PSHOME\Microsoft.PowerShell_profile.ps1):

function vcvars64 {
  cmd /K '"C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat" && powershell.exe'
}

或者,如果您使用的是 PowerShell Core:

function vcvars64 {
  cmd /K '"C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\VC\Auxiliary\Build\vcvars64.bat" && pwsh.exe'
}

然后只需在 PowerShell 中执行 vcvars64

PS C:\Users\Glavo> cl
cl: The term 'cl' is not recognized as a name of a cmdlet, function, script file, or executable program.
Check the spelling of the name, or if a path was included, verify that the path is correct and try again.
PS C:\Users\Glavo> vcvars64
**********************************************************************
** Visual Studio 2019 Developer Command Prompt v16.11.5
** Copyright (c) 2021 Microsoft Corporation
**********************************************************************
[vcvarsall.bat] Environment initialized for: 'x64'
PowerShell 7.1.5
Copyright (c) Microsoft Corporation.

https://aka.ms/powershell
Type 'help' to get help.

PS C:\Users\Glavo> cl
Microsoft (R) C/C++ Optimizing Compiler Version 19.29.30136 for x64
Copyright (C) Microsoft Corporation.  All rights reserved.

usage: cl [ option... ] filename... [ /link linkoption... ]

p
pratikpc

代替

调用批处理脚本编写环境变量以如此复杂的方式执行修改,使用 Visual Studio 2022(可能还有 2019),您可以使用以下更简单的方法

Import-Module "VS-Directory\Community\Common7\Tools\Microsoft.VisualStudio.DevShell.dll"
Enter-VsDevShell -VsInstallPath "VS-Directory\Community" -DevCmdArguments

但是,如果您希望在默认情况下为 VS 使用 64 位 MSVC 和环境,您可以使用带有 arch=64 的 DevCmdArguments

Enter-VsDevShell -VsInstallPath "VS-Directory\Community" -DevCmdArguments '-arch=x64'

P
Peter Mortensen

对于在 2020 年和 Visual Studio Code 1.41.1 中仍在苦苦挣扎的人来说,这有点离题了。

使用来自先前答案和 Internet 的所有不同代码部分,例如来自 How can I use vcvars64.bat from PowerShell? 的代码,并通过逐步的方法,我设法使以下脚本正常工作。

保存到 Visual Studio Code“settings.json”并安装了 Code Runner 扩展。

使用 Visual Studio 2015/14.0 中的 Microsoft (R) C/C++ 优化编译器版本“cl.exe”:

"code-runner.runInTerminal": true,
"code-runner.executorMap": {
  "cpp": "cd $dir; pushd \"C:\\Program Files (x86)\\Microsoft Visual Studio 14.0\\VC\"; cmd.exe /c \"call vcvarsall.bat x86_amd64 & set > %temp%\\vcvars.txt\"; Get-Content \"$env:temp\\vcvars.txt\" | Foreach-Object { if ($_ -match \"^(.*?)=(.*)$\") {   Set-Content \"env:\\$($matches[1])\" $matches[2]  }}; popd; cls; cl *.cpp; .\\\"$fileNameWithoutExt.exe\"; Write-Host -NoNewLine 'Press any key to continue...';  $null = $Host.UI.RawUI.ReadKey('NoEcho,IncludeKeyDown'); del \"$fileNameWithoutExt.exe\"; del \"$fileNameWithoutExt.obj\""}

使用 Visual Studio 2019/16.4.3 中的 Microsoft (R) C/C++ 优化编译器版本“cl.exe”:

"code-runner.runInTerminal": true,
"code-runner.executorMap": {
  "cpp": "cd $dir; pushd \"c:\\Program Files (x86)\\Microsoft Visual Studio\\2019\\Community\\VC\\Auxiliary\\Build\"; cmd.exe /c \"call vcvarsall.bat x86_amd64 & set > %temp%\\vcvars.txt\"; Get-Content \"$env:temp\\vcvars.txt\" | Foreach-Object { if ($_ -match \"^(.*?)=(.*)$\") {   Set-Content \"env:\\$($matches[1])\" $matches[2]  }}; popd; cls; cl *.cpp; .\\\"$fileNameWithoutExt.exe\"; Write-Host -NoNewLine 'Press any key to continue...';  $null = $Host.UI.RawUI.ReadKey('NoEcho,IncludeKeyDown'); del \"$fileNameWithoutExt.exe\"; del \"$fileNameWithoutExt.obj\""}

为什么 C++ 编译器进入画面?
@PeterMortensen AFAIR 更多的是演示 PowerShell 如何不仅可以从 Visual Studio 代码命令提示符启动,而且还可以在更复杂的情况下使用它,例如运行带有属性的 C++ 编译器的情况。
P
Peter Mortensen

作为 Andy S's answer 的扩展,它已针对 Visual Studio 2019 社区版进行了更新。但是,我还需要 64 位,因此这包括 -arch=amd64-host_arch=amd64 标志以及任何需要它们的人。

pushd "C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\Common7\Tools"
cmd /c "VsDevCmd.bat -arch=amd64 -host_arch=amd64&set " |
foreach {
  if ($_ -match "=") {
    $v = $_.split("="); set-item -force -path "ENV:\$($v[0])"  -value "$($v[1])"
  }
}
popd
Write-Host "`nVisual Studio 2019 Command Prompt variables set." -ForegroundColor Yellow

8
8DH

尝试 powershell 模块 posh-vsdev。它甚至似乎支持多个已安装的 Visual Studio 版本。

Install-Module -Name posh-vsdev -Scope CurrentUser

# Import the posh-vsdev module module
Import-Module posh-vsdev

# Get all installed Visual Studio instances
Get-VisualStudioVersion

# Get a specific version
Get-VisualStudioVersion -Version 14

# Use the Developer Command Prompt Environment
Use-VisualStudioEnvironment

# Restore the non-Developer environment
Reset-VisualStudioEnvironment

# Reset cache of Visual Studio instances
Reset-VisualStudioVersionCache

# Add posh-vsdev to your PowerShell profile
Add-VisualStudioEnvironmentToProfile -UseEnvironment

M
MatrixManAtYrService

我喜欢将命令传递到子 shell 中,如下所示:

cmd /c "`"${env:VS140COMNTOOLS}vsvars32.bat`" && <someCommand>"

或者,

cmd /c "`"${env:VS140COMNTOOLS}..\..\VC\vcvarsall.bat`" amd64 && <someCommand> && <someOtherCommand>"  

为什么?是什么原因?它有什么优点?请通过 editing (changing) your answer 回复,而不是在评论中(没有“编辑:”、“更新:”或类似内容 - 答案应该看起来好像是今天写的)。
抱歉@PeterMortensen,我不记得为什么上面的 hijinks 是必要的,或者为什么我更喜欢它们而不是替代方案。当我看到这个时,我唯一能想到的就是我应该早点切换到 vim。
s
sunny moon

首先,检查这个文件夹的内容:

C:/ProgramData/Microsoft/VisualStudio/Packages/_Instances/

其中会有另一个文件夹,其名称由十六进制数字组成(例如 2a7a9ed6,但对于不同的 MSVC 版本会有所不同)。我将其称为 <instance_id>

然后从 PowerShell 运行:

Import-Module 'C:\Program Files (x86)\Microsoft Visual Studio\2019\BuildTools\Common7\Tools\Microsoft.VisualStudio.DevShell.dll'; Enter-VsDevShell <instance_id> -DevCmdArguments '-arch=x64'

或者,您可以使用以下 target 创建快捷方式:

<path to your powershell.exe> -noe -c "&{Import-Module """C:\Program Files (x86)\Microsoft Visual Studio\2019\BuildTools\Common7\Tools\Microsoft.VisualStudio.DevShell.dll"""; Enter-VsDevShell <instance_id> -DevCmdArguments '-arch=x64'}"

显然,如果您需要 x86 工具集,请删除 -arch=x64


E
Emiosproj

要使用开发工具 env 启动任何最新的 PowerShell:

C:/Program Files/PowerShell/7/pwsh.exe -noe -c "&{Import-Module """C:\Program Files (x86)\Microsoft Visual Studio\2019\Community\Common7\Tools\Microsoft.VisualStudio.DevShell.dll"""; Enter-VsDevShell 5f8ef82b}" 

如果它不起作用,请通过快速启动菜单搜索 -> 右键单击属性 -> 复制目标行 -> 将 exe 路径更改为所需的 PowerShell.exe,查找适用于 VS 2019 的 Developer PowerShell

也可以为您选择的编辑器/IDE 启动它 vscode 示例:

"terminal.integrated.profiles.windows": {
"win_dev_shell_64": {
  "path": "C:/Program Files/PowerShell/7/pwsh.exe",
  "args": [
    "-noe",
    "-c",
    "&{Import-Module \"C:\\Program Files (x86)\\Microsoft Visual Studio\\2019\\Community\\Common7\\Tools\\Microsoft.VisualStudio.DevShell.dll\"; Enter-VsDevShell 5f8ef82b}"
    ]
  }
},
"terminal.integrated.defaultProfile.windows": "win_dev_shell_64",