如何从 Windows 命令行永久更新 PATH 变量?

Posted

技术标签:

【中文标题】如何从 Windows 命令行永久更新 PATH 变量?【英文标题】:How to update PATH variable permanently from Windows command line? 【发布时间】:2012-01-11 14:19:19 【问题描述】:

如果我从命令行 (cmd.exe) 执行 set PATH=%PATH%;C:\\Something\\bin,然后执行 echo %PATH%,我会看到此字符串已添加到 PATH。如果我关闭并打开命令行,则该新字符串不在 PATH 中。

我怎样才能从命令行为将来的所有进程永久更新 PATH,而不仅仅是当前进程?

我不想通过转到系统属性 → 高级 → 环境变量并在那里更新 PATH 来执行此操作。

此命令必须从 Java 应用程序执行(请参阅我的另一个 question)。

【问题讨论】:

使用 powershell,相当简单***.com/questions/714877/…。使用cmd,我不确定。您可能需要修改注册表或以某种方式引入 .net 程序集。 正如我所说,我必须在 java 应用程序中执行此操作。我以为只是执行一些使用 java 的 Runtime.getRuntime().exec("my command"); 的 cmd 命令 这能回答你的问题吗? Adding a directory to the PATH environment variable in Windows 【参考方案1】:

你可以使用:

setx PATH "%PATH%;C:\\Something\\bin"

但是,setx 会将存储的字符串截断为 1024 字节,可能会损坏 PATH。

/M 将更改HKEY_LOCAL_MACHINE 中的PATH 而不是HKEY_CURRENT_USER。换句话说,一个系统变量,而不是用户的。例如:

SETX /M PATH "%PATH%;C:\your path with spaces"

您必须记住,新的 PATH 在您当前的cmd.exe 中不可见。

但是,如果您查看注册表或使用"set p" 查找新的cmd.exe,您可以看到新值。

【讨论】:

有没有办法使用setx改变机器的路径而不是用户的路径? 从here你可以看出,在windows xp上,通过在命令末尾使用/m,不仅可以为当前登录的用户设置变量,还可以为机器设置一个变量和 7. 不过我还没试过。 运行setx命令时出现错误“默认选项不允许超过'2'时间”如何绕过它? @KilgoreCod cmets:我警告不要使用以下命令:在许多(大多数?)安装中,这些天 PATH 变量会很长 - setx 会将存储的字符串截断为 1024 字节,可能会损坏 PATH (请参阅此处的讨论superuser.com/q/812754)。 我尝试回显已经超过 1200 字节的路径。任何其他方式来代替 setx?【参考方案2】:

有关如何执行此操作的文档可以在 MSDN 上找到。关键摘录是这样的:

要以编程方式添加或修改系统环境变量,请将它们添加到 HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Session Manager\Environment 注册表项,然后广播 WM_SETTINGCHANGE 消息并将 lParam 设置为字符串“环境”。这允许应用程序(例如 shell)获取您的更新。

请注意,您的应用程序需要提升管理员权限才能修改此密钥。

您在 cmets 中表明您很乐意仅修改每个用户的环境。通过编辑 HKEY_CURRENT_USER\Environment 中的值来执行此操作。和以前一样,请确保您广播了 WM_SETTINGCHANGE 消息。

您应该能够使用 JNI 注册表类轻松地在 Java 应用程序中执行此操作。

【讨论】:

是的,使用 JNI 注册表类。一个更大的问题是您的应用程序可能没有运行提升。你知道如何让它做到这一点吗?如果您只希望应用程序的一小部分运行提升(即只是为了进行此更改),那么最简单的解决方案是一个非常简单的 C++ 应用程序来完成这项工作,用应用程序清单标记,然后作为一个单独的进程执行引发 UAC 对话。 您也可以编辑HKEY_CURRENT_USER\Environment 以避免海拔要求。 @David Heffernan 是的,只有这个东西必须提升。所以你的建议是编写 C++ 应用程序并从我的 java 应用程序中执行它?您能否提供一些示例代码或链接来说明如何执行此操作? 是的。正如大卫所说。只有你不抬高。我还应该提到这只会修改当前用户的环境。 您需要将其分离到一个单独的进程中,以便在修改系统 PATH 时仅强制执行 UAC 对话框。它只需要一个简单的 C++ 应用程序,其中包含一些注册表读取和写入,然后是一个 SendMessage。在应用清单中将requestedExecutionLevel 设置为requireAdministrator【参考方案3】:

我警告不要使用该命令

setx PATH "%PATH%;C:\Something\bin"

修改 PATH 变量,因为其实现的“特性”。在当今的许多(大多数?)安装中,变量会很长 - setx 会将存储的字符串截断为 1024 字节,可能会破坏 PATH(参见讨论 here)。

我专门注册以标记此问题,因此缺乏直接评论 12 年 5 月 2 日发布的答案的网站声誉。感谢 beresfordt 添加这样的评论

【讨论】:

【参考方案4】:

这个 Python 脚本[*] 正是这样做的:

"""
Show/Modify/Append registry env-vars (ie `PATH`) and notify Windows-applications to pickup changes.

First attempts to show/modify HKEY_LOCAL_MACHINE (all users), and 
if not accessible due to admin-rights missing, fails-back 
to HKEY_CURRENT_USER.
Write and Delete operations do not proceed to user-tree if all-users succeed.

Syntax: 
    prog                  : Print all env-vars. 
    prog  VARNAME         : Print value for VARNAME. 
    prog  VARNAME   VALUE : Set VALUE for VARNAME. 
    prog  +VARNAME  VALUE : Append VALUE in VARNAME delimeted with ';' (i.e. used for `PATH`). 
    prog  -VARNAME        : Delete env-var value. 

Note that the current command-window will not be affected, 
changes would apply only for new command-windows.
"""

import winreg
import os, sys, win32gui, win32con

def reg_key(tree, path, varname):
    return '%s\%s:%s' % (tree, path, varname) 

def reg_entry(tree, path, varname, value):
    return '%s=%s' % (reg_key(tree, path, varname), value)

def query_value(key, varname):
    value, type_id = winreg.QueryValueEx(key, varname)
    return value

def yield_all_entries(tree, path, key):
    i = 0
    while True:
        try:
            n,v,t = winreg.EnumValue(key, i)
            yield reg_entry(tree, path, n, v)
            i += 1
        except OSError:
            break ## Expected, this is how iteration ends.

def notify_windows(action, tree, path, varname, value):
    win32gui.SendMessage(win32con.HWND_BROADCAST, win32con.WM_SETTINGCHANGE, 0, 'Environment')
    print("---%s %s" % (action, reg_entry(tree, path, varname, value)), file=sys.stderr)

def manage_registry_env_vars(varname=None, value=None):
    reg_keys = [
        ('HKEY_LOCAL_MACHINE', r'SYSTEM\CurrentControlSet\Control\Session Manager\Environment'),
        ('HKEY_CURRENT_USER', r'Environment'),
    ]
    for (tree_name, path) in reg_keys:
        tree = eval('winreg.%s'%tree_name)
        try:
            with winreg.ConnectRegistry(None, tree) as reg:
                with winreg.OpenKey(reg, path, 0, winreg.KEY_ALL_ACCESS) as key:
                    if not varname:
                        for regent in yield_all_entries(tree_name, path, key):
                            print(regent)
                    else:
                        if not value:
                            if varname.startswith('-'):
                                varname = varname[1:]
                                value = query_value(key, varname)
                                winreg.DeleteValue(key, varname)
                                notify_windows("Deleted", tree_name, path, varname, value)
                                break  ## Don't propagate into user-tree.
                            else:
                                value = query_value(key, varname)
                                print(reg_entry(tree_name, path, varname, value))
                        else:
                            if varname.startswith('+'):
                                varname = varname[1:]
                                value = query_value(key, varname) + ';' + value
                            winreg.SetValueEx(key, varname, 0, winreg.REG_EXPAND_SZ, value)
                            notify_windows("Updated", tree_name, path, varname, value)
                            break  ## Don't propagate into user-tree.
        except PermissionError as ex:
            print("!!!Cannot access %s due to: %s" % 
                    (reg_key(tree_name, path, varname), ex), file=sys.stderr)
        except FileNotFoundError as ex:
            print("!!!Cannot find %s due to: %s" % 
                    (reg_key(tree_name, path, varname), ex), file=sys.stderr)

if __name__=='__main__':
    args = sys.argv
    argc = len(args)
    if argc > 3:
        print(__doc__.format(prog=args[0]), file=sys.stderr)
        sys.exit()

    manage_registry_env_vars(*args[1:])

以下是一些使用示例,假设它已保存在您当前路径某处名为setenv.py 的文件中。 请注意,在这些示例中,我没有管理员权限,因此这些更改仅影响我的本地用户的注册表树:

> REM ## Print all env-vars
> setenv.py
!!!Cannot access HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session   Manager\Environment:PATH due to: [WinError 5] Access is denied
HKEY_CURRENT_USER\Environment:PATH=...
...

> REM ## Query env-var:
> setenv.py PATH C:\foo
!!!Cannot access HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session   Manager\Environment:PATH due to: [WinError 5] Access is denied
!!!Cannot find HKEY_CURRENT_USER\Environment:PATH due to: [WinError 2] The system cannot find the file specified

> REM ## Set env-var:
> setenv.py PATH C:\foo
!!!Cannot access HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session   Manager\Environment:PATH due to: [WinError 5] Access is denied
---Set HKEY_CURRENT_USER\Environment:PATH=C:\foo

> REM ## Append env-var:
> setenv.py +PATH D:\Bar
!!!Cannot access HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session   Manager\Environment:PATH due to: [WinError 5] Access is denied
---Set HKEY_CURRENT_USER\Environment:PATH=C:\foo;D:\Bar

> REM ## Delete env-var:
> setenv.py -PATH
!!!Cannot access HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session   Manager\Environment:PATH due to: [WinError 5] Access is denied
---Deleted HKEY_CURRENT_USER\Environment:PATH

[*] 改编自:http://code.activestate.com/recipes/416087-persistent-environment-variables-on-windows/

【讨论】:

【参考方案5】:

出于参考目的,对于任何搜索如何通过代码更改路径的人,我在此网页上引用了 Delphi 程序员的一篇有用的帖子:http://www.tek-tips.com/viewthread.cfm?qid=686382

TonHu (Programmer) 22 Oct 03 17:57 I found where I read the original 发帖,在这里: http://news.jrsoftware.org/news/innosetup.isx/msg02129....

您需要的摘录如下:

您必须在 LParam 中指定字符串“Environment”。在德尔福你会 这样做:

 SendMessage(HWND_BROADCAST, WM_SETTINGCHANGE, 0, Integer(PChar('Environment')));

这是由 Jordan Russell 建议的,http://www.jrsoftware.org, (a.o.) InnoSetup 的作者,(“Inno Setup 是 Windows 程序。 Inno Setup 于 1997 年首次推出,如今可与竞争对手媲美 甚至在功能集上超过了许多商业安装程序和 稳定性。”)(我只是希望更多人使用 InnoSetup )

HTH

【讨论】:

你必须修改注册表。此外,对 Integer 的演员阵容也很差。转换为 LPARAM 以实现 64 位兼容性。 这里是例子github.com/gilligan/snesdev/blob/1253994/tools/cc65-2.13.2/…【参考方案6】:

在企业网络中,用户只有有限的访问权限并使用便携式应用程序,有以下命令行技巧:

    查询用户环境变量:reg query "HKEY_CURRENT_USER\Environment"。将 "HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session Manager\Environment" 用于 LOCAL_MACHINE。 添加新的用户环境变量:reg add "HKEY_CURRENT_USER\Environment" /v shared_dir /d "c:\shared" /t REG_SZ。对包含其他 %% 变量的路径使用 REG_EXPAND_SZ。 删除现有环境变量:reg delete "HKEY_CURRENT_USER\Environment" /v shared_dir

【讨论】:

【参考方案7】:

这个脚本 http://www.autohotkey.com/board/topic/63210-modify-system-path-gui/

包括所有必要的 Windows API 调用,可以根据您的需要进行重构。它实际上是一个 AutoHotkey GUI,可以轻松更改系统路径。需要以管理员身份运行。

【讨论】:

很棒的剧本。我使用 HotKey,但不知道如何或需要做什么才能将脚本添加到其中。您能提供帮助、提供链接或解释需要做什么吗?

以上是关于如何从 Windows 命令行永久更新 PATH 变量?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 CMAKE 从命令行在 Windows 上构建 x86 和/或 x64?

windows7 下,在CMD命令模式下,如何添加永久路由? 麻烦说详细一点。

Windows 非服务器版本永久关闭命令行“快速编辑模式”

DOS永久设置系统环境变量-WMIC

windows设置临时环境变量path

在windows中设置环境变量PATH