[新手上路]批处理新手入门导读[视频教程]批处理基础视频教程[视频教程]VBS基础视频教程[批处理精品]批处理版照片整理器
[批处理精品]纯批处理备份&还原驱动[批处理精品]CMD命令50条不能说的秘密[在线下载]第三方命令行工具[在线帮助]VBScript / JScript 在线参考
返回列表 发帖

[系统相关] [己解決]批处理命令 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

回复 33# 77七
經測試確實可能,感謝幫大忙.

TOP

按13楼方法修改,试试能不能用,已上传到网盘。https://f.ws59.cn/f/ebmzeuacj08 复制链接到浏览器打开
bat小白,请多指教!谢谢!

TOP

回复 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 的情况,使其配对,然后使用预处理方式带出变量。
bat小白,请多指教!谢谢!

TOP

本帖最后由 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

TOP

回复 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. )
复制代码
可以得到想要的結果.

TOP

回复 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
复制代码
bat小白,请多指教!谢谢!

TOP

回复 23# dreamer


   能否分享一下最终的代码?比如以15楼代码为例。如何能通过powershell取得新窗口的变量?
bat小白,请多指教!谢谢!

TOP

回复 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 ::"
复制代码
bat小白,请多指教!谢谢!

TOP

本帖最后由 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七的參與討論,沒有您的討論恐怕我也找不出原因.

TOP

回复 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裡顯示為空值(它在舊窗口被賦值)

TOP

回复 21# dreamer


   call 15楼脚本后
旧窗口
  1. a
  2. ECHO 处于关闭状态。
复制代码


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


新窗口设置的变量不能回传到旧窗口的。旧窗口的变量可以在新窗口继承。
bat小白,请多指教!谢谢!

TOP

回复 20# newswan
這個方式真是太棒了,不管要什麼變數只要echo出來就有了.

TOP

回复 19# 77七
對,那主要是測試是否在開新窗口後是否能取得變數

TOP

回复 15# 77七
直接call腳本會得到二個窗口,
新窗口
  1. a
  2. ECHO 已關閉。
  3. 請按任意鍵繼續 . . .
复制代码
舊窗口
  1. a
  2. b
  3. 請按任意鍵繼續 . . .
复制代码
新窗口不帶a變數,我理解您的意思了,原來是我們各自有不同的關注,我的關注是新開窗口後產生的變數是否能取得,您則是指舊窗口產生的變數是否能延續.
測試結果如您所說變數不能延續,也如我所說可以取得新窗口變數.

TOP

返回列表