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