0

0

C 中使用 goto 进行错误处理

PHPz

PHPz

发布时间:2024-07-10 11:25:46

|

591人浏览过

|

来源于dev.to

转载

c 中使用 goto 进行错误处理

最近,Python Brasil 邮件列表上开始了关于使用异常的原因的讨论。有一次,一位非常有能力的参与者评论了通过函数返回来处理错误是多么困难,就像在 C 中一样。

当你有一个复杂的算法时,每个可能失败的操作都需要一系列的 if 来检查操作是否成功。如果操作失败,您需要恢复之前的所有操作以退出算法,而不改变程序的状态。

让我们看一个例子。假设我有以下结构来表示数组:

typedef 结构体 {
    整数大小;
    整数*数组;
} array_t;

现在,我将编写一个函数,从文本文件中读取要放置在这些数组之一中的元素数量,然后读取元素本身。该函数还将分配数组结构和数组本身。问题是这个函数很容易出错,因为我们可能会失败:

  • 打开给定的文件;
  • 分配结构体;
  • 由于输入/输出错误或文件结尾而读取给定文件中的元素数量;
  • 分配内存来存储要读取的元素;
  • 由于输入/输出错误或文件结尾而读取其中一个元素。

很复杂吧?请注意,如果我们设法打开文件但无法分配结构,则必须关闭该文件;如果我们设法打开文件并分配结构体,但无法从文件中读取元素数量,则必须释放结构体并关闭文件;等等。因此,如果我们检查所有错误并采用在出现错误时返回 NULL 的传统,我们的函数将如下所示:

array_t *readarray(const char *文件名) {
    文件*文件;
    array_t *数组;
    整数我;

    文件 = fopen(文件名, "r");
    如果(文件== NULL)返回NULL;

    数组= malloc(sizeof(array_t));
    如果(数组== NULL){
        fclose(文件);
        返回空值;
    }

    if (fscanf(文件, "%d", &(数组->大小)) == EOF) {
        自由(数组);
        fclose(文件);
        返回空值;
    }

    数组->数组 = malloc(sizeof(int) * 数组->大小);
    if (数组->数组 == NULL) {
        自由(数组);
        fclose(文件);
        返回空值;
    }

    for (i = 0; i < 数组->大小; i++) {
        if (fscanf(文件, "%d", 数组->数组 + i) == EOF) {
            自由(数组->数组);
            自由(数组);
            fclose(文件);
            返回空值;
        }
    }
    返回数组;
}

确实,相当费力,而且有很多重复的代码……

但是请注意,上面的代码中有两种情况

冰舟分类信息系统
冰舟分类信息系统

冰舟分类信息系统说明:本次更新修改了部分错误,增加了自定义标签管理,这个版本后所有页面均可调用一个标签,大大的提升了效率使用前请先 运行 install.asp 文件进行安装程序!!安装时请填写好 ACCESS文件目录以及ACCESS文件名,请确保填写的信息与FTP上的完全吻合,否则会出错!!安装玩后请,务必删除 install.asp 文件! 本2.1版本含有强大的模板编辑功能,且初始模板均为D

下载
  1. 其中,当我有两个操作要恢复时,我需要先恢复最后执行的一个,然后再恢复前一个。例如,当释放结构体和整数数组时,我需要先释放整数数组,然后释放结构体。如果我先释放结构体,以后可能无法释放数组。
  2. 在其他情况下,顺序并不重要。例如,如果我要释放结构体并关闭文件,那么执行此操作的顺序并不重要。这意味着我也可以先恢复最后执行的操作,然后恢复第一个操作。

这有什么意义?嗯,在实践中,我从未见过必须先恢复第一个执行的操作,然后恢复第二个,依此类推的情况。这意味着,当执行操作 a()、b()、c() 等时,恢复它们的“自然”方法是以相反的顺序调用恢复函数,如下所示:

a();
b();
C();
/* ... */
revert_c();
恢复b();
恢复a();

现在窍门来了。在上面的代码中,在每个操作之后,我们都会放置一个 if 来检查它是否失败。如果失败,将执行 goto 到上次成功操作的恢复函数:

a();
if (failed_a()) 转到 FAILED_A;
b();
if (failed_b()) 转到 FAILED_B;
C();
if (failed_c()) 转到 FAILED_C;
/* ... */
revert_c();
失败_C:
恢复b();
失败_B:
恢复a();
失败_A:
返回;

如果a()失败,算法返回;如果 b() 失败,算法将转到 FAILED_B:,恢复 a() 并返回;如果 c() 失败,算法将转到 FAILED_C,恢复 b(),恢复 a(),然后返回。你能看到图案吗?

如果我们将此模式应用于 readarray() 函数,结果将类似于:

array_t *readarray(const char *文件名) {
    文件*文件;
    array_t *数组;
    整数我;

    文件 = fopen(文件名, "r");
    如果(文件== NULL)转到FILE_ERROR;

    数组= malloc(sizeof(array_t));
    如果(数组== NULL)转到ARRAY_ALLOC_ERROR;

    if (fscanf(文件, "%d", &(数组->大小)) == EOF)
        转到SIZE_READ_ERROR;

    数组->数组 = malloc(sizeof(int) * 数组->大小);
    if (array->array == NULL) 转到 ARRAY_ARRAY_ALLOC_ERROR;

    for (i = 0; i < 数组->大小; i++) {
        if (fscanf(文件, "%d", 数组->数组 + i) == EOF)
            转到 ARRAY_CONTENT_READ_ERROR;
    }
    返回数组;

    ARRAY_CONTENT_READ_ERROR:
    自由(数组->数组);
    ARRAY_ARRAY_ALLOC_ERROR:
    大小读取错误:
    自由(数组);
    ARRAY_ALLOC_ERROR:
    fclose(文件);
    文件错误:
    返回空值;
}

这种模式有什么优点?嗯,它减少了操作反转代码的重复,并将错误处理代码与功能逻辑分离。事实上,虽然我认为异常是最好的现代错误处理方法,但对于本地错误处理(在函数本身内部),我发现这种方法更实用。

(这篇文章是 Tratamento de error em C com goto 的翻译,最初发表于 Suspensão de Descrença。)

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

763

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

639

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

764

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

619

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1285

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

549

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

709

2023.08.11

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 5.6万人学习

Django 教程
Django 教程

共28课时 | 3.3万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号