C++ 的 goto 语句在大多数时间是被打入冷宫的,就因为它容易使代码晦涩难读。同时,break 和 continue 又经常在多重循环中黔驴技穷——不能越层控制循环。

  相比之下,Java 就兼顾这两种情况,将 goto 彻底禁用,而标签则限制其只能用于循环之前,以此增加 break 和 continue 的功能,使其能越层控制循环 [参阅:《带标签的 break 和 continue (Java)》]

  Java 尚能如此,为何不在 C++ 中用标签和 goto 语句模仿 Java 来增强 break 和 continue 的功能呢?

  于是照 Java 代码写了一份 C++ 的试验样例,结果第一次试验失败。代码如下:

  1. int main(int argc, char* argv[])  
  2. ...{  
  3.     cout << "[begin>" << endl;  
  4. continuei:  
  5.     for (int i = 0; i < 10; i++) ...{  
  6.         for (int j = 0; j < 5; j++) ...{  
  7.             if ((i + j) % 5 == 0) ...{  
  8.                 cout << "continue i" <<endl;  
  9.                 goto continuei;  
  10.             }  
  11.             cout << i << ", " << j << endl;  
  12.         }  
  13.     }  
  14.     cout << "<end]" << endl;  

这样做的结果是死循环。每次 goto continuei 之后,就会重新执行第一个循环,所以造成了死循环。看来,只能把标签定义在紧接着循环结束符(},完成花括号)之前,于是又起一例。这次测试成功。代码如下:

  1. int main(int argc, char* argv[])  
  2. ...{  
  3.     cout << "[begin>" << endl;  
  4.     for (int i = 0; i < 10; i++) ...{  
  5.         for (int j = 0; j < 5; j++) ...{  
  6.             if ((i + j) % 5 == 0) ...{  
  7.                 cout << "continue i" <<endl;  
  8.                 goto continuei;  
  9.             }  
  10.             cout << i << ", " << j << endl;  
  11.         }  
  12. continuei: continue;  
  13.     }  
  14.     cout << "<end]" << endl;  

不过,这里如果 break 和 continue 同时出现的话,continue 应该放在前面。因为,如果循环过程中没有触发 goto 语句,这个循环在执行到本次循环结束的时候,是应该继续下一次循环的。continue 定义在前面,就会先执行到它,也就会路过后面后 break 继续下一次循环。返之,如果把 break 放在 continue 前面,则会跳过后面的 continue,跳出循环——这就违背了初衷。还是兴趣个例子:

  1. int main(int argc, char* argv[])  
  2. ...{  
  3.     cout << "[begin>" << endl;  
  4.     for (int i = 0; i < 10; i++) ...{  
  5.         for (int j = 0; j < 5; j++) ...{  
  6.             if ((i + j) % 5 == 0) ...{  
  7.                 cout << "continue i" <<endl;  
  8.                 goto continuei;  
  9.             }  
  10.             if (i == 7 && j == 1) ...{  
  11.                 cout << "break i" << endl;  
  12.                 goto breaki;  
  13.             }  
  14.             cout << i << ", " << j << endl;  
  15.         }  
  16. continuei: continue;  
  17. breaki: break;  
  18.     }  
  19.     cout << "<end]" << endl;  

最后还是要说明一下,这个方法要慎用,因为多重循环本身就应该慎用的。多数需多重循环的代码,都可以拆分成每个循环一个函数来替代。这样做会让代码更易读,但是也会降低程序的执行效率。