CMD中goto语句会中断for循环特性详解

这个程序里面由于用到了上篇文章中所说的字符串切割,而用到了Goto强制跳转语句

但是在程序中使用的时候却发现一个错误,当把这个字符切割的代码段如果直接作为非嵌套语句执行正常

但是一旦放到for循环的复合语句中就会发现for循环只执行一次之后for循环就退出了,而且临时变量%%x的值就变成空了

换句话说就是goto跳转和目标地:labe不能同时出现在一个for语句中

例如下示例:

for /f %%i in ("abc") do (
  :show
  echo %%i & pause
  goto :show
)

直接看代码含义很简单就是一直去显示abc这个字符串,但是实际执行结果却是:

只有第一次循环正确的执行的我们想要的结果,显示了abc,但是后面的所有都是显示%i

换句话说,当goto跳转后%i失去了原本for循环的临时遍历变量的作用,成了一个普通的字符串

至于为什么会这样我是这样理解的:

因为goto语句是无条件强制跳转,并且无视程序流程,即不管目标标签地是否是复合语句都会强制去顺着标签地向下执行,也许标签地是主流程语句,也许是其他if的复合语句块,也许是与goto是同属一个复合语句块,如果是跳转到外侧或其他语句块,那么当前for循环肯定是要退出的。

但是CMD是边解释边执行,他没有办法也没有义务去检查goto的标签所在地是在哪,从理论上来讲,如果跳转到主流程没有什么问题,如果跳转到另一复合语句块中肯定目标语句块的临时变量(%i)肯定是无效的,因为根本不知道从什么位置直接跳转过来的,根本没有执行主循环语句,何谈临时变量,临时变量是由主循环语句赋予的。

唯独goto和标签同属一个语句块时临时变量应该起效。但是作为边解释边执行的解释器,压根无法预知下一条根本没有输入的语句的内容。

所以(结论):

在for语句中一旦遇到goto,则cmd认为这是break了,不管你跳转到哪,当前for循环肯定是停止了,那么由for循环赋予的临时变量(%i)也就随之无效了。在刚才的例子中虽然跳转完成了还是在for语句中,但cmd已经认为for循环已经退出,这是独立语句了,而不是复合语句

解决方法:

把要进行goto的语句块写成一个函数,写成一个独立模块,再在for中用Call去调用,就正常了

上面的例子改一下:

for /f %%i in ("abc") do (
  call :show %%i
)

:show
echo %1 & pause
goto :show

结果,如我们想要的,一直去显示abc字符串

—— 原文发表于2012-3-11 08:57

原文地址:https://www.cnblogs.com/xxcanghai/p/4584137.html