标题: [系统相关] [己解決]批处理命令 call 無法傳值? [打印本页]
作者: dreamer 时间: 2024-6-8 15:41 标题: [己解決]批处理命令 call 無法傳值?
本帖最后由 dreamer 于 2024-6-10 08:42 编辑
問題描述:下方批次檔是將Office 零售轉大量授權的工具,由於要自動化所以製作一個呼叫他的批次檔- Call "%~pd0C2R-R2V-AIO.cmd"
- Echo %_msg%&Rem C2R-R2V-AIO檔的變數
复制代码
在C2R-R2V-AIO的1481行是離開批次的處理段落,有試著在1487行前加上复制代码
但卻沒有效果,這個問題困擾了我一週,最終還是沒有找到答案,目前推測可能是過多的exit /b 跟 goto 所以造成非預期的現象,但還是想了解真正的原因而不是推測,希望有高手能給出更合理的分析,謝謝。
這個問題目前暫時用doskey+findstr解決(思路是將結果放到doskey後再用findstr判斷結果是否為finished=Finished),由於代碼過長貼不下源代碼如下
來源:https://github.com/abbodi1406/C2R-R2V-AIO
作者: 77七 时间: 2024-6-8 17:17
脚本太复杂,没看明白意思,大概理了下退出的代码
第1是 104行之前 多个 goto :e_ 后退出,这之前没有定义msg。可以忽略。
第2是 104行之后 第117行,if 语句两个call 其中定义了msg,并在141行退出。故可以试试在141行之前添加一行- endlocal & set _msg=%msg%
复制代码
作者: aloha20200628 时间: 2024-6-8 18:17
未见源代码,但若在复合语块,如 for...do() 或 if...() 中用 endlocal&set _msg=%_msg% 这类 ‘续命’ 的玩法,会被 ‘预处理’ 提前毙掉的...
作者: newswan 时间: 2024-6-8 19:37
本帖最后由 newswan 于 2024-6-8 20:14 编辑
- Call "%~pd0C2R-R2V-AIO.cmd" _msg
- Echo %_msg%&Rem C2R-R2V-AIO檔的變數
复制代码
- :E_Exit
- endlocal&Set %1=%msg%
复制代码
作者: dreamer 时间: 2024-6-9 06:06
回复 2# 77七
感謝指點,追蹤了下141行复制代码
它似乎只會有空值
作者: dreamer 时间: 2024-6-9 06:07
回复 3# aloha20200628
感謝指點,這個是我沒吸收到的,謝謝。
作者: dreamer 时间: 2024-6-9 06:19
回复 4# newswan
感謝回覆,感覺他會是空值,實作也是空值.
作者: newswan 时间: 2024-6-9 08:40
用 for 获取输出- pushd "%~dp0"
-
- for /f "usebackq tokens=1,* delims= " %%a in (` C2R-R2V-AIO.cmd `) do (
- if "%%a" == "msg" (set _msg_=%%b)
- )
-
- popd
复制代码
原文件 1481复制代码
作者: 77七 时间: 2024-6-9 11:44
本帖最后由 77七 于 2024-6-9 13:09 编辑
回复 5# dreamer
方法1将msg结果重定向到文本
140行左右添加行
- @echo off
- >>"%~dp0#.txt" echo,%msg%
- @exit /b
复制代码
1482行左右添加行
- :E_Exit
- >>"%~dp0#.txt" echo,%msg%
- if %_Debug% EQU 1 goto :eof
- echo.
- echo Press any key to exit.
复制代码
使用以下脚本调用
- @echo off
- cd /d "%~dp0"
- rem 把两个脚本放在同一目录下。或者自行修改两个脚本的 #.txt 路径
- del "%~dp0#.txt" 2>nul
- start "" "%~dp0C2R-R2V-AIO.cmd"
- :loop
- if not exist "%~dp0#.txt" (
- timeout 1
- goto :loop
- )
- set _msg=
- for /f "useback delims=" %%a in ("%~dp0#.txt") do (
- set _msg=%%a
- )
- if defined _msg (
- echo %_msg%
- ) else (
- echo 原脚本未定义msg
- )
- del "%~dp0#.txt" 2>nul
- pause
复制代码
方法2
可能需要将原脚本 36,41行两个start cmd.exe 及相关部分移植到调用脚本中,这样可以直接 call 调用,有点复杂
我是这样认为的,不知道对不对
作者: dreamer 时间: 2024-6-9 13:30
回复 8# newswan
又學到一招了!真心感謝,這個方式可以取得C2R-R2V-AIO所有的輸出文字.感覺在這個基礎上,可以將所有需要的函數用Echo輸出而不用考慮setlocal造成的問題,太棒了!
作者: dreamer 时间: 2024-6-9 13:42
回复 9# 77七
之前有測試寫入文件確實可行,想說寫入文件那干脆寫入到doskey 裡會不會更好一些.我的作法是在C2R-R2V-AIO 的:E_Exit後加上- If "%msg%" EQU "Finished" Doskey Finished=%msg%
复制代码
然後在用findstr找出符合的字串,因為Finished是唯一條件,找到就成立找不到就失敗,所以用以下內容去判斷- Doskey /macros|findstr /B /E /C:"finished=Finished">Nul&&Echo 完成
复制代码
作者: 77七 时间: 2024-6-9 13:53
回复 11# dreamer
- start %SystemRoot%\Sysnative\cmd.exe /c ""!_cmdf!" -wow "
复制代码
这行代码作用不是打开一个新窗口?打开的新窗口,call 已经获取不到其变量了吧
作者: 77七 时间: 2024-6-9 14:33
本帖最后由 77七 于 2024-6-9 14:51 编辑
回复 11# dreamer
可以试着用以下脚本调用
脚本已经修改,刚才逻辑不对- @echo off
- if exist "%SystemRoot%\Sysnative\cmd.exe" (
- call %SystemRoot%\Sysnative\cmd.exe /c "%~dp0C2R-R2V-AIO.cmd"
- ) else (
- if exist "%SystemRoot%\SysArm32\cmd.exe" (
- if /i %PROCESSOR_ARCHITECTURE%==AMD64 (
- call %SystemRoot%\SysArm32\cmd.exe /c "%~dp0C2R-R2V-AIO.cmd"
- ) else call "%~dp0C2R-R2V-AIO.cmd"
- ) else call "%~dp0C2R-R2V-AIO.cmd"
- )
- pause
复制代码
删除原脚本第25-43行,注意脚本中的路径,然后按照2楼方法。试试能不能取得msg值。(当然,2楼已经说明,msg存在空值可能)
本人认知、能力有限,可以试一下。
作者: dreamer 时间: 2024-6-9 15:19
回复 12# 77七
33行有定義,它再次呼叫自身,所以應該還是可以取到值.复制代码
作者: 77七 时间: 2024-6-9 15:43
回复 14# dreamer
无论他执行哪个脚本,只要打开了新窗口,新窗口可以继承原窗口的变量,而不能回传。
可以简单写个脚本验证
- @echo off
-
- if "%~1" equ "" (
- title 旧
- set a=a
- start "" "%~f0" xx
- ) else (
- title 新
- set b=b
- )
- echo %a%
- echo %b%
- pause
复制代码
作者: dreamer 时间: 2024-6-9 15:52
回复 14# dreamer
試了下確實還是可以取得變數內容,1.cmd 呼叫用复制代码
2.cmd的內容是- @echo off
- setlocal
- %1Call :aaa
- endlocal&Set AA=參數一
- goto :EOF
- :aaa
- endlocal&Set AA=參數二
- start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码
作者: dreamer 时间: 2024-6-9 16:04
回复 2# 77七
分析了下,141行的退出似乎只在啟用除錯模式下且條件符合時在141退出,141行前多個條件若有不符時會跳至相應標記- goto :E_WMI
- goto :E_WSH
- goto :E_VBS
- goto :E_PWS
- 最後都會到
- goto :E_Exit
复制代码
它確實有二個出口,不過能取得%msg%是Finished就足夠了,沒有Finished就是失敗.
作者: dreamer 时间: 2024-6-9 16:10
回复 9# 77七
試了下還是沒有傳回值,因為該批次要求系統管理員權限進行操作,我使用管理員權限操作,是因為這個因素嗎?
作者: 77七 时间: 2024-6-9 16:31
回复 16# dreamer
直接call 15楼脚本,遇见什么错误了吗?16楼脚本,打开新窗口后有设置什么不同的变量吗?最后执行的不都goto :eof 之前的 set aa=参数一?
作者: newswan 时间: 2024-6-9 19:46
C2R-R2V-AIO.cmd 很长 ,里面 start cmd call
不要去折腾里面的流程
powershell 获取 bat 的 echo 输出- $batout = Invoke-Expression -Command " cmd.exe /c 'C2R-R2V-AIO.cmd' "
复制代码
更简单些
作者: dreamer 时间: 2024-6-10 00:21
回复 15# 77七
直接call腳本會得到二個窗口,
新窗口- a
- ECHO 已關閉。
- 請按任意鍵繼續 . . .
复制代码
舊窗口复制代码
新窗口不帶a變數,我理解您的意思了,原來是我們各自有不同的關注,我的關注是新開窗口後產生的變數是否能取得,您則是指舊窗口產生的變數是否能延續.
測試結果如您所說變數不能延續,也如我所說可以取得新窗口變數.
作者: dreamer 时间: 2024-6-10 00:23
回复 19# 77七
對,那主要是測試是否在開新窗口後是否能取得變數
作者: dreamer 时间: 2024-6-10 00:26
回复 20# newswan
這個方式真是太棒了,不管要什麼變數只要echo出來就有了.
作者: 77七 时间: 2024-6-10 01:08
回复 21# dreamer
call 15楼脚本后
旧窗口
复制代码
新窗口
复制代码
新窗口设置的变量不能回传到旧窗口的。旧窗口的变量可以在新窗口继承。
作者: dreamer 时间: 2024-6-10 05:24
回复 16# dreamer
回复 24# 77七
24樓說的沒錯,是我改繁體字時誤改新為舊,舊為新了,不過我的測試結果與您不同,應該是用start 及cmd.exe /c的差異
如果將16樓的2.cmd改成- @echo off
- setlocal
- %1Call :aaa
- endlocal&Set AA=參數一
- goto :EOF
- :aaa
- endlocal&Set BB=參數二
- start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码
BB變數在1.cmd裡顯示為空值(它在舊窗口被賦值)
作者: dreamer 时间: 2024-6-10 08:01
本帖最后由 dreamer 于 2024-6-10 08:04 编辑
回复 25# dreamer
哈哈,我想C2R-R2V-AIO取不到值的原因找到了
1.cmd呼叫用- call 2.cmd
- Echo AAA:%AA%
- Echo BBB:%BB%
- pause
复制代码
2.cmd被呼叫的內容- @echo off
- setlocal
- %1Call :aaa
- endlocal&Set BB=%BB%
- endlocal&Set AA=參數一
- Goto :EOF
- :aaa
- endlocal&Set BB=參數二
- start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码
若是將2.cmd第4行及第五行互換結果會不一樣,BB會取不到值,就如同C2R-R2V-AIO的情況一樣,看來是與setlocal套嵌層數有關, endlocal要符合當前層才能取出值.由於無法明確知道到底call 了多少次,用newswan的方式或許簡單些.
感謝77七的參與討論,沒有您的討論恐怕我也找不出原因.
作者: 77七 时间: 2024-6-10 11:30
回复 25# dreamer
仅仅测试,不需要写 很多开关区域变量
- @echo off
- %1Call :aaa
- Set AA=參數一
- goto :EOF
- :aaa
- Set BB=參數二
- start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码
是因为两次endlocal ,第二次endlocal 没有给BB变量 通过预处理的方式逃脱区域变量限制。
- @echo off
- setlocal
- %1Call :aaa
- endlocal&Set AA=參數一&set BB=參數二
- goto :EOF
- :aaa
- endlocal&Set BB=參數二
- start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码
作者: 77七 时间: 2024-6-10 11:32
回复 23# dreamer
能否分享一下最终的代码?比如以15楼代码为例。如何能通过powershell取得新窗口的变量?
作者: 77七 时间: 2024-6-10 11:35
回复 26# dreamer
这个问题,我没注意到,如果不知道多少次setlocal,可以通过代码,多次endlocal,并带出变量。- @echo off
- setlocal
- setlocal
- setlocal
- set a=a
- for /f "delims=" %%a in ("%a%") do (
- for /l %%l in (1,1,32) do endlocal 2>nul
- set _a=%%a
- )
- echo %_a%
- pause
复制代码
作者: dreamer 时间: 2024-6-10 15:12
回复 28# 77七
我沒有使用powershell 一樣用cmd 處理,因為C2R-R2V-AIO如果成功會輸出一行Finished,參考newswan的方式稍加變化- for /f "usebackq tokens=* delims= " %%a in (` C2R-R2V-AIO.cmd `) do (
- if "%%a" == "Finished" (set _msg_=%%a)
- )
复制代码
可以得到想要的結果.
作者: dreamer 时间: 2024-6-10 15:51
本帖最后由 dreamer 于 2024-6-10 16:01 编辑
回复 28# 77七
剛誤解您的意思了,改用call 就能回傳值
ps.顯示上只能看到"新",要看到"舊"需要加pause,不過a是由舊賦值也可證先跑舊再跑新- @echo off
- if "%1" equ "" (
- title 舊
- set a=a
- call "%~f0" ###
- ) else (
- title 新
- set b=b
- )
- echo %a%
- echo %b%
- pause
复制代码
start "" "%0"
或
"%0"
都是另開線程,call跑同一線程,所以可以取得下層(新開窗)變數.
以15樓例來說,跑到start 就沒if的事了,它接著執行echo a,b 所以舊只會有a
以上面的例子來說if 還要等call做完事才會去echo a,b 所以會有a,b
作者: 77七 时间: 2024-6-10 16:07
回复 30# dreamer
好几层楼都在讨论,新开的窗口的变量不能回传。虽然问题已经解决,我觉得是个特例,因为 在执行 C2R-R2V-AIO.cmd 中,以下判断均失败的原因,直接到了 set 这一行。- set "_cmdf=%~f0"
- if exist "%SystemRoot%\Sysnative\cmd.exe" if not defined _rel1 (
- setlocal EnableDelayedExpansion
- start %SystemRoot%\Sysnative\cmd.exe /c ""!_cmdf!" -wow "
- exit /b
- )
- if exist "%SystemRoot%\SysArm32\cmd.exe" if /i %PROCESSOR_ARCHITECTURE%==AMD64 if not defined _rel2 (
- setlocal EnableDelayedExpansion
- start %SystemRoot%\SysArm32\cmd.exe /c ""!_cmdf!" -arm "
- exit /b
- )
- set "SysPath=%SystemRoot%\System32"
复制代码
特例中没有新开窗口,所有得到了期望的结果。
还有一点我觉得需要注意第136行 已经重定向了 call的子函数的输出,如果脚本不加改动,并且由136行执行子函数, for /f 是不会读取到这部分的。或者从 log文件能不能找到执行的结果呢?
- @call :Begin >"!_log!_tmp.log"
复制代码
如果想让脚本更加通用,可以尝试13楼方法,并采用上此次回帖中,for /f +多次endlocal 带出变量,或者直接 改善C2R-R2V-AIO.cmd 脚本,多处setlocal 缺少endlocal 的情况,使其配对,然后使用预处理方式带出变量。
作者: 77七 时间: 2024-6-10 16:39
按13楼方法修改,试试能不能用,已上传到网盘。https://f.ws59.cn/f/ebmzeuacj08 复制链接到浏览器打开
作者: dreamer 时间: 2024-6-12 02:45
回复 33# 77七
經測試確實可能,感謝幫大忙.
欢迎光临 批处理之家 (http://bbs.bathome.net/) |
Powered by Discuz! 7.2 |