官方帮助里有一段管理员权限运行自身的代码,但是那段代码有4个问题。
1是不能指定ahk版本。
2是不能降权运行(管理员权限降为普通权限)。
3是会丢失传入的参数。
4是每次复制粘贴一遍代码很烦。
本函数解决了以上4个问题。
使用方法:
- 将以下代码保存为文件“RunWith.ahk”。
- 放入本地库、用户库、标准库其中任意一个里(例如 “X:\我的电脑\Documents\AutoHotkey\Lib”)。
; 强制自身进程以 管理员权限 或 普通权限 或 ANSI 或 U32 或 U64 版本运行。
; 例1: runwith("admin","u32") 强制自身以 u32 + 管理员权限 运行。
; 例2: runwith("","ansi") 强制自身以 ansi 版本运行(权限不变)。
; 例3: runwith("normal") 强制自身以 普通权限 运行(版本不变)。
RunWith(RunAsAdmin:="Default", ANSI_U32_U64:="Default")
{
; 格式化预期的模式
switch, RunAsAdmin
{
case "Normal","Standard","No","0": RunAsAdmin:=0
case "Admin","Yes","1": RunAsAdmin:=1
case "Default": RunAsAdmin:=A_IsAdmin
default: RunAsAdmin:=A_IsAdmin
}
switch, ANSI_U32_U64
{
case "A32","ANSI","A": ANSI_U32_U64:="AutoHotkeyA32.exe"
case "U32","X32","32": ANSI_U32_U64:="AutoHotkeyU32.exe"
case "U64","X64","64": ANSI_U32_U64:="AutoHotkeyU64.exe"
case "Default": ANSI_U32_U64:="AutoHotkey.exe"
default: ANSI_U32_U64:="AutoHotkey.exe"
}
; 获取传递给 “.ahk” 的用户参数(不是 /restart 之类传递给 “.exe” 的开关参数)
for k, v in A_Args
{
if (RunAsAdmin=1)
{
; 转义所有的引号与转义符号
v:=StrReplace(v, "\", "\\")
v:=StrReplace(v, """", "\""")
; 无论参数中是否有空格,都给参数两边加上引号
; Run 的内引号是 "
ScriptParameters .= (ScriptParameters="") ? """" v """" : A_Space """" v """"
}
else
{
; 转义所有的引号与转义符号
; 注意要转义两次 Run 和 RunAs.exe
v:=StrReplace(v, "\", "\\")
v:=StrReplace(v, """", "\""")
v:=StrReplace(v, "\", "\\")
v:=StrReplace(v, """", "\""")
; 无论参数中是否有空格,都给参数两边加上引号
; RunAs.exe 的内引号是 \"
ScriptParameters .= (ScriptParameters="") ? "\""" v "\""" : A_Space "\""" v "\"""
}
}
; 判断当前 exe 是什么版本
if (!A_IsUnicode)
RunningEXE:="AutoHotkeyA32.exe"
else if (A_PtrSize=4)
RunningEXE:="AutoHotkeyU32.exe"
else if (A_PtrSize=8)
RunningEXE:="AutoHotkeyU64.exe"
; 运行模式与预期相同,则直接返回。 ANSI_U32_U64="AutoHotkey.exe" 代表不对 ahk 版本做要求。
if (A_IsAdmin=RunAsAdmin and (ANSI_U32_U64="AutoHotkey.exe" or ANSI_U32_U64=RunningEXE))
return
; 如果当前已经是使用 /restart 参数重启的进程,则报错避免反复重启导致死循环。
else if (RegExMatch(DllCall("GetCommandLine", "str"), " /restart(?!\S)"))
{
预期权限:=(RunAsAdmin=1) ? "管理员权限" : "普通权限"
当前权限:=(A_IsAdmin=1) ? "管理员权限" : "普通权限"
ErrorMessage=
(LTrim
预期使用: %ANSI_U32_U64%
当前使用: %RunningEXE%
预期权限: %预期权限%
当前权限: %当前权限%
程序即将退出。
)
MsgBox 0x40030, 运行状态与预期不一致, %ErrorMessage%
ExitApp
}
else
{
; 获取 AutoHotkey.exe 的路径
SplitPath, A_AhkPath, , Dir
if (RunAsAdmin=0)
{
; 强制普通权限运行
switch, A_IsCompiled
{
; %A_ScriptFullPath% 必须加引号,否则含空格的路径会被截断。%ScriptParameters% 必须不加引号,因为构造时已经加了。
; 工作目录不用单独指定,默认使用 A_WorkingDir 。
case, "1": Run, RunAs.exe /trustlevel:0x20000 "\"%A_ScriptFullPath%\" /restart %ScriptParameters%",, Hide
default: Run, RunAs.exe /trustlevel:0x20000 "\"%Dir%\%ANSI_U32_U64%\" /restart \"%A_ScriptFullPath%\" %ScriptParameters%",, Hide
}
}
else
{
; 强制管理员权限运行
switch, A_IsCompiled
{
; %A_ScriptFullPath% 必须加引号,否则含空格的路径会被截断。%ScriptParameters% 必须不加引号,因为构造时已经加了。
; 工作目录不用单独指定,默认使用 A_WorkingDir 。
case, "1": Run, *RunAs "%A_ScriptFullPath%" /restart %ScriptParameters%
default: Run, *RunAs "%Dir%\%ANSI_U32_U64%" /restart "%A_ScriptFullPath%" %ScriptParameters%
}
}
ExitApp
}
}
测试代码:(使用 Unicode32 版本并以管理员权限运行自身)
RunWith("admin",32)
当前AHK版本 := (!A_IsUnicode) ? "ANSI" : (A_PtrSize=4) ? "Unicode 32" : "Unicode 64"
当前权限 := (A_IsAdmin=1) ? "管理员权限" : "普通权限"
MsgBox, % "当前AHK版本: " 当前AHK版本 "`n`n当前权限: " 当前权限
很不错,不过这个必须放ahk安装目录是不是不方便?
随便放啊 只是一个函数…… 放公共Lib目录只是为了方便所有代码都可以使用而已。
把 RunWith() 函数代码复制到自己的AHK脚本文件中 或者
放公共Lib目录 或者
放自己的AHK脚本所在文件夹然后再 #Include RunWith.ahk
都是可以的,
我认为函数和对象(数组、属性、方法)是最方便易用的代码组织形式了。
非常感谢 RunWith() 函数的作者 空
这个 RunWith() 函数解决了我的一大难题
为以后的.ahk脚本添加以管理员权限运行
RegWrite, REG_SZ, HKCUSoftwareMicrosoftWindows NTCurrentVersionAppCompatFlagsLayers,%A_AhkPath%, ~ RUNASADMIN
RegWrite, REG_SZ, HKCRSoftwareMicrosoftWindows NTCurrentVersionAppCompatFlagsLayers,%A_AhkPath%, ~ RUNASADMIN
和这个
RunWith() 函数一起用,双管齐下,彻底解决脚本取得管理员权限的问题
好东西!
有时候总是测试不通过,还以为是自己的代码写错了,最后还是要靠这个脚本才行。
非常感谢作者写出这么神奇的代码
这个只支持V1吗? 不支持V2呢? 改一下支持v2好吗?
@空 以管理员身份运行程序的话应当登录管理员,请问,普通用户没有系统管理员的密码应当不能以管理员身份运行程序,是不是?以登录管理员的运行环境的话应该就可以以管理员身份运行程序了,是不是?不知道我理解的对不对,特来请教。 有没有办法在不登录管理员并且不知道管理员密码的情况下以管理员身份运行程序或者提高权限再运行程序,谢谢。