batch-file - 为什么其他文件夹路径也使用 SetX 添加到系统 PATH 而不仅仅是指定的文件夹路径?

标签 batch-file registry

我有一个批处理文件,我使用 system("name.bat") 从 C++ 调用它。在该批处理文件中,我试图读取注册表项的值。从 C++ 调用批处理文件会导致 set KEY_NAME=HKEY_LOCAL_MACHINE\stuff 失败。

但是,当我直接运行批处理文件(双击它)时,它运行良好。不确定我做错了什么。

批处理文件:

set KEY_NAME=HKEY_LOCAL_MACHINE\SOFTWARE\Ansoft\Designer\2014.0\Desktop
set VALUE_NAME=InstallationDirectory
REG QUERY %KEY_NAME% /v %VALUE_NAME%

C++ 文件:

int main(void)
{
    system("CALL C:\\HFSS\\setup_vars.bat");
    return 0;
}

更新 1:

我发现 key 实际上在 64 位注册表中,而我正在将我的 C++ 解决方案构建为 32 位。修复该问题后,它发现注册表项正常。

现在我在将该路径添加到我的 PATH 变量时遇到问题。它不是创建系统变量,而是创建用户变量 PATH 并将其添加到那里。

从命令行运行。

代码:

set KEY_NAME=HKLM\SOFTWARE\Ansoft\Designer\2014.0\Desktop\
set VALUE_NAME=InstallationDirectory

FOR /F "usebackq skip=1 tokens=1,2*" %%A IN (`REG QUERY %KEY_NAME% /v %VALUE_NAME%`) DO (
   set ValueName=%%A
   set ValueType=%%B
   set ValueValue=%%C
)

if defined ValueName (
   @echo Value Value = %ValueValue%
) else (
   @echo %KEY_NAME%\%VALUE_NAME% not found.
)

:: Set PATH Variable
set path_str=%PATH%
set addPath=%ValueValue%;

echo %addPath%
echo %ValueValue%

echo %PATH%| find /i "%addPath%">NUL

if NOT ERRORLEVEL 1 (
   SETX PATH "%PATH%
) else (
   SETX PATH "%PATH%;%addPath%;" /M
)

更新 2:

我移动了选项/M 的位置,它现在添加到正确的 PATH 变量中。

但是,当我这样做时,它不止一次(3 次)添加了 PATH,然后它还添加了 visual studio amd64 文件夹的路径。

我不确定为什么会这样。

最佳答案

Windows 会为新进程创建一个进程的整个环境表的副本。因此,在您的 C++ 应用程序启动时,您的应用程序会从父进程、Windows 资源管理器或您的 Visual Studio 获取环境表,包括 PATH。并且此 PATH 是在批处理文件开始时为 cmd.exe 复制的。

考虑到从 Windows 桌面到批处理文件的整个进程树,已经为 PATH 制作了许多副本,并且一些进程可能会在其本地 PATH 就像 Visual Studio 所做的那样,甚至已经从 PATH 中删除了路径。

您现在使用 SETX PATH "%PATH% 所做的是将已由进程树中的父进程修改的 PATH 的本地副本完全附加到系统 PATH 而不检查重复路径。

更好的方法是丢弃所有使用 PATH 的本地副本的代码,而是读取系统 PATH 的值,检查您要添加的路径是否不是已经在系统 PATH 中,如果不是这种情况,请使用 setx 将要添加的路径附加到系统 PATH

这应该在不扩展系统 PATH 中的环境变量的情况下完成,例如 %SystemRoot%\System32C:\Windows\System32 .


更新

这是在 Windows 7 x64 和 Windows XP x86 上测试任务所需的批处理代码。

@echo off
setlocal EnableExtensions DisableDelayedExpansion
set "KeyName=HKLM\SOFTWARE\Ansoft\Designer\2014.0\Desktop"
set "ValueName=InstallationDirectory"
for /F "skip=2 tokens=1,2*" %%N in ('%SystemRoot%\System32\reg.exe query "%KeyName%" /v "%ValueName%" 2^>nul') do (
    if /I "%%N" == "%ValueName%" (
        set "PathToAdd=%%P"
        if defined PathToAdd goto GetSystemPath
    )
)
echo Error: Could not find non-empty value "%ValueName%" under key
echo        %KeyName%
echo/
endlocal
pause
goto :EOF

:GetSystemPath
for /F "skip=2 tokens=1,2*" %%N in ('%SystemRoot%\System32\reg.exe query "HKLM\System\CurrentControlSet\Control\Session Manager\Environment" /v "Path" 2^>nul') do (
    if /I "%%N" == "Path" (
        set "SystemPath=%%P"
        if defined SystemPath goto CheckPath
    )
)
echo Error: System environment variable PATH not found with a non-empty value.
echo/
endlocal
pause
goto :EOF

:CheckPath
setlocal EnableDelayedExpansion
rem The folder path to add must contain \ (backslash) as directory
rem separator and not / (slash) and should not end with a backslash.
set "PathToAdd=%PathToAdd:/=\%"
if "%PathToAdd:~-1%" == "\" set "PathToAdd=%PathToAdd:~0,-1%"
set "Separator="
if not "!SystemPath:~-1!" == ";" set "Separator=;"
set "PathCheck=!SystemPath!%Separator%"
if "!PathCheck:%PathToAdd%;=!" == "!PathCheck!" (
    set "PathToSet=!SystemPath!%Separator%!PathToAdd!"
    set "UseSetx=1"
    if not "!PathToSet:~1024,1!" == "" set "UseSetx="
    if not exist %SystemRoot%\System32\setx.exe set "UseSetx="
    if defined UseSetx (
        %SystemRoot%\System32\setx.exe Path "!PathToSet!" /M >nul
    ) else (
        set "ValueType=REG_EXPAND_SZ"
        if "!PathToSet:%%=!" == "!PathToSet!" set "ValueType=REG_SZ"
        %SystemRoot%\System32\reg.exe ADD "HKLM\System\CurrentControlSet\Control\Session Manager\Environment" /f /v Path /t !ValueType! /d "!PathToSet!" >nul
    )
)
endlocal
endlocal

上面的批处理代码使用简单的不区分大小写的字符串替换和区分大小写的字符串比较来检查系统 PATH 中是否已经存在要追加的文件夹路径。仅当众所周知之前添加文件夹路径的方式并且用户在此期间未在 PATH 中修改此文件夹路径时,此方法才有效。有关检查 PATH 是否包含文件夹路径的更安全方法,请参阅 How to check if directory exists in %PATH%? 上的答案。由 Dave Benham 撰写.

注意 1:命令 setx 在 Windows XP 上默认不可用。

注意 2: 命令 setx 将长度超过 1024 个字符的值截断为 1024 个字符。

因此,如果 setx 不可用或有新路径值,批处理文件使用命令 reg 替换 Windows 注册表中的 system PATH对于 setx 来说太长了。使用 reg 的缺点是 WM_SETTINGCHANGE消息不会发送到所有顶级窗口,通知作为 Windows 桌面运行的 Windows 资源管理器和其他应用程序有关系统环境变量的更改。因此,用户必须重新启动 Windows,最好始终在更改持久存储的 Windows 系统环境变量上的某些内容时完成。

批处理脚本使用 PATH 进行了测试,该路径当前包含一个带感叹号的文件夹路径,并且文件夹路径用双引号括起来,只有当文件夹路径包含分号时才需要。

要了解使用的命令及其工作原理,请打开命令提示符窗口,在其中执行以下命令,并仔细阅读为每个命令显示的所有帮助页面。

  • echo/?
  • endlocal/?
  • for/?
  • goto/?
  • 如果/?
  • 暂停/?
  • reg/?reg add/?reg query/?
  • 设置/?
  • setlocal/?
  • setx/?

关于batch-file - 为什么其他文件夹路径也使用 SetX 添加到系统 PATH 而不仅仅是指定的文件夹路径?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25915767/

相关文章:

batch-file - 您可以创建一个在批处理文件中包含变量的变量吗?

batch-file - 您可以从命令行运行 AIR 应用程序吗?

batch-file - “@echo off”和“@echo%off”之间的区别?

visual-studio-2012 - 如何制作可移植visual studio 2012? (使用外部驱动器和脚本)

C++程序需要一个文件关联

C++ 从 64 位应用程序读取 SOFTWARE\WOW6432 中的注册表项

windows - 为什么不需要 `call` 从管道中涉及的被调用批处理脚本返回?

for-loop - 如何使用forfiles命令批量使用多个命令?

java - 使用 Java 在 Windows 上查找用户的“我的文档”文件夹的正确方法?

java - 有没有办法在 Windows 下使用 java.util.Preferences 而无需使用注册表作为后端?