GOTO 语句

C/C++ 的 goto 语句用来在一个函数内进行任意跳转,用起来也是很方便。示例如下:

int a() {
int x = 0, sum = 0;
L1: if (x < 10) {
x++;
sum += x;
goto L1;
}
return sum;
}

只需要在函数内部某处加一个标签(Label),通过 goto <label> 即可直接跳转。

但是由于 goto 的跳转比较原始,没有结构化,当程序比较庞大的时候调试起来就比较麻烦,代码可读性会大大降低,因此 E.W.Dijikstra 在 1965 年提出结构化程序设计来规避这种错误,并使代码更易读。很多时候我们尽量避免使用 goto 语句,转而使用其他的结构化方式。

尽管如此,也有一些地方使用 goto 语句相比结构化的方式更加高效。以下给出若干例子。

跳出多层循环

跳出循环的语句是 break,跳过一次循环的语句是 continue,但它们仅能跳出/跳过一层循环。当需要跳出多层循环时,可以考虑使用 goto 语句:

LOOP: for (...) // loop 1
for (...) // loop 2
for (...) // loop 3
if (...)
goto LOOP;

此处的 goto 语句从第三层循环跳到第一层循环,相当于在第一层循环下执行了 continue 语句。而如果要用结构化的方式来实现,往往要借助条件变量。尽管如此,仍然需要一次次地跳出当前循环:

int flag = 0;
for (...) // loop 1
for (...) // loop 2
for (...) { // loop 3
if (...) {
flag = 1;
break;
}
}
if (flag) break;
}
}

可以看到,这次除了引入一个 flag 变量外,用了两个 break 语句才实现了等价功能。

很多高级语言都不支持 goto 语句以防止可能遇到的问题,比如 Java。但是 Java 支持用 break <label>continue <label> 来支持多层循环的跳出与跳过,比如该 goto 语句示例在 Java 中的等价写法是把 goto 关键字换成 continue。因此,这种用法不应该被嫌弃。

循环首次部分跳过

如果把一个循环体分为前后两部分 A 和 B,那么循环展开后的执行情况为:

ABABABAB...ABAB

但是,很多情况下我们可能要求跳过第一次运行的 A 或者最后一次运行的 B,那么实际的运行情况为:

BABABAB...ABAB

最经典的例子就是打印一个数组且要求结尾没有空格。此时可以考虑使用 goto 语句:

void print_array(int arr[], int n) {
int i = 0;
if (i<n) goto PNT;
for (; i<n; i++) {
putchar(' '); // A
PNT:
printf("%d", arr[i]); // B
}
puts(); // 换行
}

此时循环展开就完美按照 BABABA...BA 的顺序执行以打印数组。如果不用 goto 语句,那么可以考虑两种方案:

  1. 给 A 部分加上条件判断是否为第一次循环。但这样的话,每次循环时都会进行判断,造成不必要的判断过程;
  2. 直接把 B 部分代码放到循环前面。但是当 B 部分代码特别多时,该方法会导致严重的代码重复。尽管可以使用函数将 B 部分封装,但是函数调用也会带来额外开销。

相比之下,这种情况使用 goto 语句完全不影响循环体代码,效率更高。

注意,这种情况下要求循环体的循环条件语句尽量简单(大多数情况下是简单的,比如此处的 i<n),因为对于 for、while 等循环会在进入循环前判断一次,因此这种写法通常会在 goto 前加上 if 语句进行同样的循环判断。如果判断语句过长过复杂,同样会造成一定的代码冗余。当然,对于 do-while 循环,其循环条件是在每次循环后判断,此时不必在 goto 跳转前进行判断。

另外,该方法在编写代码时,如果要达到 ABABAB...ABA 的效果,即跳过最后一次循环的后半部分,应该直接在循环体内将 A、B 的顺序调换,并用 goto 语句跳过第一次循环的前半部分。


原文地址:https://www.cnblogs.com/RainbowC0/p/18765900 ,未经作者许可禁止转载。

C/C++ GOTO妙用的更多相关文章

  1. C语言中do...while(0)的妙用-避免goto

    使用goto的优雅并避免结构的混乱 将要跳转到的语句用do{-}while(0) 包起来就可以. reference #defien N 10 bool Execute() { // 分配资源 int ...

  2. C语言中do...while(0)的妙用(转载)

    转载来自:C语言中do...while(0)的妙用,感谢分享. 在linux内核代码中,经常看到do...while(0)的宏,do...while(0)有很多作用,下面举出几个: 1.避免goto语 ...

  3. 求1+2+…+n,要求不能使用乘除法、for、while、if、else、s witch、case 等关键字以及条件判断语句(A?B:C)和不用循环/goto/递归输出1~100的10种写法

    来源:据说是某一年某个公司的面试题 题目:求1+2+…+n, 要求不能使用乘除法.for.while.if.else.s witch.case 等关键字以及条件判断语句(A?B:C) 分析:这题本来很 ...

  4. C语言 goto语句

    /* goto语句 */ #include <stdio.h> #include <stdlib.h> #include <string.h> /* goto语句也 ...

  5. 【C++基础 03】do...while(0)妙用

    我的主题是,有时候知道一些细节会让你写出更好的代码. ============================================ 之前学coocs2d-x的时候,发现有非常多do...w ...

  6. windows脚本(VBS)之cmd命令行的妙用

    windows脚本(VBS)之cmd命令行的妙用 (2009-08-06 13:40:55) 转载▼ 标签: 脚本 cmd 命令行 vbs js 简单 公式 windows it 分类: 计算机 脚本 ...

  7. 【转】do...while(0)的妙用

    前言 今天无意中看到这个标题,因为好奇就点进去了,不错,又学习啦... 具体内容: 1. do...while(0)消除goto语句: 2 宏定义中的do...while(0): 参考 1. 原链接_ ...

  8. 臭名远扬之 goto 语句

    C 语言自学之 goto 语句 Dome1:以下程序实现从控制台输出1-10,使用goto语句,实现当输出完3之后跳出循环体. 1 #include <stdio.h> 2 3 int m ...

  9. 【CSS进阶】伪元素的妙用--单标签之美

    最近在研读 <CSS SECRET>(CSS揭秘)这本大作,对 CSS 有了更深层次的理解,折腾了下面这个项目: CSS3奇思妙想 -- Demo (请用 Chrome 浏览器打开,非常值 ...

  10. angular2系列教程(十)两种启动方法、两个路由服务、引用类型和单例模式的妙用

    今天我们要讲的是ng2的路由系统. 例子

随机推荐

  1. Mac触控板设置以及使用

    Mac 触控板体验是非常好的,很多同学甚至直接用触控板代替鼠标操作,但是默认设置中有一些功能是没有开启的,需要手动配置. 本文就来说说 如何更改 Mac 触控板默认设置,让触控板变得更高效. 一.启用 ...

  2. Qt编写地图综合应用44-悬浮工具条

    一.前言 百度地图内置了悬浮工具条,可以自行开启,包括离线地图也可以开启,用到了DrawingManager这个库,鼠标绘制工具条库,提供鼠标绘制点.线.面.多边形(矩形.圆)的编辑工具条的开源代码库 ...

  3. Qt通用方法及类库10

    函数名 //获取保存的文件 static QString getSaveName(const QString &filter, QString defaultDir = QCoreApplic ...

  4. Qt编写安防视频监控系统39-onvif图片参数

    一.前言 通过onvif来调整图片的Brightness(亮度).ColorSaturation(色彩饱和度).Contrast(饱和度)这三个参数,可以实时观测到监控画面对应的变化,比如讲亮度Bri ...

  5. [转]C#从MySQL数据库中读取

    实现了数据库的建表.存储数据的功能后,还需要实现数据库的读取,综合查资料后发现有两种发发比较好; 一.如需要界面操作,需要将数据表格在界面上显示出来的话,需要使用DataGrid控件. 基本操作流程: ...

  6. 推荐 5 个 火火火火 的CMS开源.Net项目

    下面推荐5个基于.NetCore开发的CMS开源项目. 一.OrchardCore 基于ASP.NET Core 构建的.模块化和多租户应用程序框架,采用文档数据库,非常高性能,跨平台的系统. 1.跨 ...

  7. 阿里IM技术分享(九):深度揭密RocketMQ在钉钉IM系统中的应用实践

    本文由钉钉技术专家尹启绣分享,有修订和重新排版. 1.引言 短短的几年时间,钉钉便迅速成为一款国民级应用,发展速度堪称迅猛. IM作为钉钉最核心的功能,每天需要支持海量企业用户的沟通,同时还通过 Pa ...

  8. 【漏洞分析】20250105-SorraStaking:奖励金额计算错误,每次取款都有大收益

    背景信息 2024-12-21 11:58:11 (UTC) 准备交易:https://app.blocksec.com/explorer/tx/eth/0x72a252277e30ea6a37d2d ...

  9. w3cschool-MyBatis-Plus 插件

    https://www.w3cschool.cn/mybatis_plus/mybatis_plus-udwn3mgc.html MyBatis-Plus(简称 MP)是一个 MyBatis的增强工具 ...

  10. mysql数据库主从同步I/O问题修复

    mysql数据库主从同步I/O问题,下面介绍比较靠谱的修复方法. 主节点IP:10.99.202.25,从节点IP:10.99.202.26,修复步骤如下: 1,查看主库repl账号访问权限 mysq ...