Board logo

标题: [系统相关] [己解決]批处理命令 call 無法傳值? [打印本页]

作者: dreamer    时间: 2024-6-8 15:41     标题: [己解決]批处理命令 call 無法傳值?

本帖最后由 dreamer 于 2024-6-10 08:42 编辑

問題描述:下方批次檔是將Office 零售轉大量授權的工具,由於要自動化所以製作一個呼叫他的批次檔
  1. Call "%~pd0C2R-R2V-AIO.cmd"
  2. Echo %_msg%&Rem C2R-R2V-AIO檔的變數
复制代码
在C2R-R2V-AIO的1481行是離開批次的處理段落,有試著在1487行前加上
  1. endlocal&Set _msg=%_msg%
复制代码
但卻沒有效果,這個問題困擾了我一週,最終還是沒有找到答案,目前推測可能是過多的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行之前添加一行
  1. 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 编辑
  1. Call "%~pd0C2R-R2V-AIO.cmd" _msg
  2. Echo %_msg%&Rem C2R-R2V-AIO檔的變數
复制代码
  1. :E_Exit
  2. endlocal&Set %1=%msg%
复制代码

作者: dreamer    时间: 2024-6-9 06:06

回复 2# 77七
感謝指點,追蹤了下141行
  1. endlocal&set _msg=%msg%
复制代码
它似乎只會有空值
作者: dreamer    时间: 2024-6-9 06:07

回复 3# aloha20200628
感謝指點,這個是我沒吸收到的,謝謝。
作者: dreamer    时间: 2024-6-9 06:19

回复 4# newswan

感謝回覆,感覺他會是空值,實作也是空值.
作者: newswan    时间: 2024-6-9 08:40

用 for 获取输出
  1. pushd "%~dp0"
  2. for /f "usebackq tokens=1,* delims= " %%a in (` C2R-R2V-AIO.cmd `) do (
  3. if "%%a" == "msg" (set _msg_=%%b)
  4. )
  5. popd
复制代码
原文件 1481
  1. :E_Exit
  2. echo msg %msg%
复制代码

作者: 77七    时间: 2024-6-9 11:44

本帖最后由 77七 于 2024-6-9 13:09 编辑

回复 5# dreamer


  方法1将msg结果重定向到文本
140行左右添加行
  1. @echo off
  2. >>"%~dp0#.txt" echo,%msg%
  3. @exit /b
复制代码


1482行左右添加行
  1. :E_Exit
  2. >>"%~dp0#.txt" echo,%msg%
  3. if %_Debug% EQU 1 goto :eof
  4. echo.
  5. echo Press any key to exit.
复制代码


使用以下脚本调用
  1. @echo off
  2. cd /d "%~dp0"
  3. rem 把两个脚本放在同一目录下。或者自行修改两个脚本的 #.txt 路径
  4. del "%~dp0#.txt" 2>nul
  5. start "" "%~dp0C2R-R2V-AIO.cmd"
  6. :loop
  7. if not exist "%~dp0#.txt" (
  8. timeout 1
  9. goto :loop
  10. )
  11. set _msg=
  12. for /f "useback delims=" %%a in ("%~dp0#.txt") do (
  13. set _msg=%%a
  14. )
  15. if defined _msg (
  16. echo %_msg%
  17. ) else (
  18. echo 原脚本未定义msg
  19. )
  20. del "%~dp0#.txt" 2>nul
  21. 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後加上
  1. If "%msg%" EQU "Finished" Doskey Finished=%msg%
复制代码
然後在用findstr找出符合的字串,因為Finished是唯一條件,找到就成立找不到就失敗,所以用以下內容去判斷
  1. Doskey /macros|findstr /B /E /C:"finished=Finished">Nul&&Echo 完成
复制代码

作者: 77七    时间: 2024-6-9 13:53

回复 11# dreamer


  
  1. start %SystemRoot%\Sysnative\cmd.exe /c ""!_cmdf!" -wow "
复制代码
这行代码作用不是打开一个新窗口?打开的新窗口,call 已经获取不到其变量了吧
作者: 77七    时间: 2024-6-9 14:33

本帖最后由 77七 于 2024-6-9 14:51 编辑

回复 11# dreamer


   可以试着用以下脚本调用
脚本已经修改,刚才逻辑不对
  1. @echo off
  2. if exist "%SystemRoot%\Sysnative\cmd.exe" (
  3. call %SystemRoot%\Sysnative\cmd.exe /c "%~dp0C2R-R2V-AIO.cmd"
  4. ) else (
  5. if exist "%SystemRoot%\SysArm32\cmd.exe" (
  6. if /i %PROCESSOR_ARCHITECTURE%==AMD64 (
  7. call %SystemRoot%\SysArm32\cmd.exe /c "%~dp0C2R-R2V-AIO.cmd"
  8. ) else call "%~dp0C2R-R2V-AIO.cmd"
  9. ) else call "%~dp0C2R-R2V-AIO.cmd"
  10. )
  11. pause
复制代码



删除原脚本第25-43行,注意脚本中的路径,然后按照2楼方法。试试能不能取得msg值。(当然,2楼已经说明,msg存在空值可能)
本人认知、能力有限,可以试一下。
作者: dreamer    时间: 2024-6-9 15:19

回复 12# 77七
33行有定義,它再次呼叫自身,所以應該還是可以取到值.
  1. set "_cmdf=%~f0"
复制代码

作者: 77七    时间: 2024-6-9 15:43

回复 14# dreamer


   
无论他执行哪个脚本,只要打开了新窗口,新窗口可以继承原窗口的变量,而不能回传。
可以简单写个脚本验证
  1. @echo off
  2. if "%~1" equ "" (
  3. title 旧
  4. set a=a
  5. start "" "%~f0" xx
  6. ) else (
  7. title 新
  8. set b=b
  9. )
  10. echo %a%
  11. echo %b%
  12. pause
复制代码

作者: dreamer    时间: 2024-6-9 15:52

回复 14# dreamer
試了下確實還是可以取得變數內容,1.cmd 呼叫用
  1. call 2.cmd
  2. Echo AAA:%AA%
复制代码
2.cmd的內容是
  1. @echo off
  2. setlocal
  3. %1Call :aaa
  4. endlocal&Set AA=參數一
  5. goto :EOF
  6. :aaa
  7. endlocal&Set AA=參數二
  8. start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码

作者: dreamer    时间: 2024-6-9 16:04

回复 2# 77七
分析了下,141行的退出似乎只在啟用除錯模式下且條件符合時在141退出,141行前多個條件若有不符時會跳至相應標記
  1. goto :E_WMI
  2. goto :E_WSH
  3. goto :E_VBS
  4. goto :E_PWS
  5. 最後都會到
  6. 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 输出
  1. $batout = Invoke-Expression -Command " cmd.exe /c 'C2R-R2V-AIO.cmd' "
复制代码
更简单些
作者: dreamer    时间: 2024-6-10 00:21

回复 15# 77七
直接call腳本會得到二個窗口,
新窗口
  1. a
  2. ECHO 已關閉。
  3. 請按任意鍵繼續 . . .
复制代码
舊窗口
  1. a
  2. b
  3. 請按任意鍵繼續 . . .
复制代码
新窗口不帶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楼脚本后
旧窗口
  1. a
  2. ECHO 处于关闭状态。
复制代码


新窗口
  1. a
  2. b
复制代码


新窗口设置的变量不能回传到旧窗口的。旧窗口的变量可以在新窗口继承。
作者: dreamer    时间: 2024-6-10 05:24

回复 16# dreamer
回复 24# 77七
24樓說的沒錯,是我改繁體字時誤改新為舊,舊為新了,不過我的測試結果與您不同,應該是用start 及cmd.exe /c的差異
如果將16樓的2.cmd改成
  1. @echo off
  2. setlocal
  3. %1Call :aaa
  4. endlocal&Set AA=參數一
  5. goto :EOF
  6. :aaa
  7. endlocal&Set BB=參數二
  8. 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呼叫用
  1. call 2.cmd
  2. Echo AAA:%AA%
  3. Echo BBB:%BB%
  4. pause
复制代码
2.cmd被呼叫的內容
  1. @echo off
  2. setlocal
  3. %1Call :aaa
  4. endlocal&Set BB=%BB%
  5. endlocal&Set AA=參數一
  6. Goto :EOF
  7. :aaa
  8. endlocal&Set BB=參數二
  9. 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


   
仅仅测试,不需要写 很多开关区域变量
  1. @echo off
  2. %1Call :aaa
  3. Set AA=參數一
  4. goto :EOF
  5. :aaa
  6. Set BB=參數二
  7. start "" "%SystemRoot%\system32\cmd.exe" /c "%~f0 ::"
复制代码

BB變數在1.cmd裡顯示為空值
是因为两次endlocal ,第二次endlocal 没有给BB变量 通过预处理的方式逃脱区域变量限制。

  1. @echo off
  2. setlocal
  3. %1Call :aaa
  4. endlocal&Set AA=參數一&set BB=參數二
  5. goto :EOF
  6. :aaa
  7. endlocal&Set BB=參數二
  8. 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,并带出变量。
  1. @echo off
  2. setlocal
  3. setlocal
  4. setlocal
  5. set a=a
  6. for /f "delims=" %%a in ("%a%") do (
  7. for /l %%l in (1,1,32) do endlocal 2>nul
  8. set _a=%%a
  9. )
  10. echo %_a%
  11. pause
复制代码

作者: dreamer    时间: 2024-6-10 15:12

回复 28# 77七
我沒有使用powershell 一樣用cmd 處理,因為C2R-R2V-AIO如果成功會輸出一行Finished,參考newswan的方式稍加變化
  1. for /f "usebackq tokens=* delims= " %%a in (` C2R-R2V-AIO.cmd `) do (
  2. if "%%a" == "Finished" (set _msg_=%%a)
  3. )
复制代码
可以得到想要的結果.
作者: dreamer    时间: 2024-6-10 15:51

本帖最后由 dreamer 于 2024-6-10 16:01 编辑

回复 28# 77七
剛誤解您的意思了,改用call 就能回傳值
ps.顯示上只能看到"新",要看到"舊"需要加pause,不過a是由舊賦值也可證先跑舊再跑新
  1. @echo off
  2. if "%1" equ "" (
  3. title 舊
  4. set a=a
  5. call "%~f0" ###
  6. ) else (
  7. title 新
  8. set b=b
  9. )
  10. echo %a%
  11. echo %b%
  12. 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 这一行。
  1. set "_cmdf=%~f0"
  2. if exist "%SystemRoot%\Sysnative\cmd.exe" if not defined _rel1 (
  3. setlocal EnableDelayedExpansion
  4. start %SystemRoot%\Sysnative\cmd.exe /c ""!_cmdf!" -wow "
  5. exit /b
  6. )
  7. if exist "%SystemRoot%\SysArm32\cmd.exe" if /i %PROCESSOR_ARCHITECTURE%==AMD64 if not defined _rel2 (
  8. setlocal EnableDelayedExpansion
  9. start %SystemRoot%\SysArm32\cmd.exe /c ""!_cmdf!" -arm "
  10. exit /b
  11. )
  12. set "SysPath=%SystemRoot%\System32"
复制代码


特例中没有新开窗口,所有得到了期望的结果。
还有一点我觉得需要注意第136行 已经重定向了 call的子函数的输出,如果脚本不加改动,并且由136行执行子函数, for /f 是不会读取到这部分的。或者从 log文件能不能找到执行的结果呢?
  1. @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://www.bathome.net/) Powered by Discuz! 7.2