活动介绍

C语言文件操作、错误处理及UNIX系统调用详解

立即解锁
发布时间: 2025-08-22 00:24:53 阅读量: 3 订阅数: 6
PDF

C语言编程基础与实践

# C语言文件操作、错误处理及UNIX系统调用详解 ## 1. 文件指针释放与错误处理 在进行文件操作时,当文件指针不再需要时,释放它们是个不错的做法。同时,对于输出文件使用 `fclose` 还有另一个原因,它会刷新 `putc` 收集输出的缓冲区。当程序正常终止时,会为每个打开的文件自动调用 `fclose`。若不需要 `stdin` 和 `stdout`,也可以将它们关闭,还能通过库函数 `freopen` 重新分配。 ### 1.1 错误处理 在处理文件操作时,错误处理至关重要。以 `cat` 程序为例,其错误处理并不理想。若某个文件因某种原因无法访问,诊断信息会打印在拼接输出的末尾。若输出到屏幕,这或许可以接受,但如果输出到文件或通过管道传输到另一个程序,就不太合适了。 为了更好地处理这种情况,引入了第二个输出流 `stderr`,它和 `stdin`、`stdout` 一样被分配给程序。即使标准输出被重定向,写入 `stderr` 的输出通常也会显示在屏幕上。 下面是修改后的 `cat` 程序,将错误信息写入标准错误: ```c #include <stdio.h> /* cat: concatenate files, version 2 */ main(int argc, char *argv[]) { FILE *fp; void filecopy(FILE *, FILE *); char *prog = argv[0]; /* program name for errors */ if (argc == 1 ) /* no args; copy standard input */ filecopy(stdin, stdout); else while (--argc > 0) if ((fp = fopen(*++argv, "r")) == NULL) { fprintf(stderr, "%s: can't open %s\n", prog, *argv); exit(1); } else { filecopy(fp, stdout); fclose(fp); } if (ferror(stdout)) { fprintf(stderr, "%s: error writing stdout\n", prog); exit(2); } exit(0); } ``` 该程序通过两种方式发出错误信号: - 由 `fprintf` 产生的诊断输出会发送到 `stderr`,这样它会显示在屏幕上,而不会消失在管道或输出文件中。消息中包含了程序名(来自 `argv[0]`),若该程序与其他程序一起使用,就能识别错误的来源。 - 程序使用标准库函数 `exit`,调用时会终止程序执行。`exit` 的参数可供调用该程序的任何进程使用,因此使用该程序作为子进程的另一个程序可以测试其成功或失败。通常,返回值 0 表示一切正常,非零值通常表示异常情况。`exit` 会为每个打开的输出文件调用 `fclose`,以刷新任何缓冲输出。 在 `main` 函数中,`return expr` 等同于 `exit(expr)`。`exit` 的优势在于它可以从其他函数调用,并且可以使用像第 5 章中的模式搜索程序找到对它的调用。 函数 `ferror` 若在流 `fp` 上发生错误,则返回非零值: ```c int ferror(FILE *fp) ``` 虽然输出错误很少见,但确实会发生(例如,磁盘已满),因此生产程序也应该检查这一点。 函数 `feof(FILE *)` 与 `ferror` 类似,若指定文件已到达文件末尾,则返回非零值: ```c int feof(FILE *fp) ``` 在小型示例程序中,我们通常不担心退出状态,但任何严肃的程序都应注意返回合理、有用的状态值。 ### 1.2 行输入和输出 标准库提供了输入输出例程 `fgets`,它类似于我们在前面章节中使用的 `getline` 函数: ```c char *fgets(char *line, int maxline, FILE *fp) ``` `fgets` 从文件 `fp` 读取下一个输入行(包括换行符)到字符数组 `line` 中,最多读取 `maxline - 1` 个字符。结果行以 `'\0'` 结尾。通常 `fgets` 返回 `line`,在文件末尾或出错时返回 `NULL`。(我们的 `getline` 返回行长度,这是一个更有用的值,零表示文件结束。) 对于输出,函数 `fputs` 将一个字符串(不一定包含换行符)写入文件: ```c int fputs(char *line, FILE *fp) ``` 若发生错误,它返回 `EOF`,否则返回非负值。 库函数 `gets` 和 `puts` 与 `fgets` 和 `fputs` 类似,但操作的是 `stdin` 和 `stdout`。令人困惑的是,`gets` 会删除终止的 `'\n'`,而 `puts` 会添加它。 下面是 `fgets` 和 `fputs` 的实现: ```c /* fgets: get at most n chars from iop */ char *fgets(char *s, int n, FILE *iop) { register int c; register char *cs; cs = s; while (--n > 0 && (c = getc(iop)) != EOF) if ((*cs++ = c) == '\n') break; *cs = '\0'; return (c == EOF && cs == s) ? NULL : s; } /* fputs: put string s on file iop */ int fputs(char *s, FILE *iop) { int c; while (c = *s++) putc(c, iop); return ferror(iop) ? EOF : 0; } ``` 标准为 `ferror` 和 `fputs` 指定了不同的返回值,原因并不明显。 使用 `fgets` 实现 `getline` 很容易: ```c /* getline: read a line, return length */ int getline(char *line, int max) { if (fgets(line, max, stdin) == NULL) return 0; else return strlen(line); } ``` ### 1.3 杂项函数 标准库提供了各种各样的函数,下面简要介绍一些最有用的函数。 #### 1.3.1 字符串操作 | 函数名 | 功能 | | ---- | ---- | | `strcat(s,t)` | 将 `t` 连接到 `s` 的末尾 | | `strncat(s,t,n)` | 将 `t` 的 `n` 个字符连接到 `s` 的末尾 | | `strcmp(s,t)` | 若 `s < t` 返回负值,`s == t` 返回零,`s > t` 返回正值 | | `strncmp(s,t,n)` | 与 `strcmp` 类似,但只比较前 `n` 个字符 | | `strcpy(s,t)` | 将 `t` 复制到 `s` | | `strncpy(s,t,n)` | 最多将 `t` 的 `n` 个字符复制到 `s` | | `strlen(s)` | 返回 `s` 的长度 | | `strchr(s,c)` | 返回指向 `s` 中第一个 `c` 的指针,若不存在则返回 `NULL` | | `strrchr(s,c)` | 返回指向 `s` 中最后一个 `c` 的指针,若不存在则返回 `NULL` | #### 1.3.2 字符类测试和转换 | 函数名 | 功能 | | ---- | ---- | | `isalpha(c)` | 若 `c` 是字母,则返回非零值,否则返回 0 | | `isupper(c)` | 若 `c` 是大写字母,则返回非零值,否则返回 0 | | `islower(c)` | 若 `c` 是小写字母,则返回非零值,否则返回 0 | | `isdigit(c)` | 若 `c` 是数字,则返回非零值,否则返回 0 | | `isalnum(c)` | 若 `isalpha(c)` 或 `isdigit(c)` 为真,则返回非零值,否则返回 0 | | `isspace(c)` | 若 `c` 是空格、制表符、换行符、回车符、换页符、垂直制表符,则返回非零值,否则返回 0 | | `toupper(c)` | 将 `c` 转换为大写字母 | | `tolower(c)` | 将 `c` 转换为小写字母 | #### 1.3.3 Ungetc 标准库提供了一个功能受限的 `ungetch` 函数版本,称为 `ungetc`: ```c int ungetc(int c, FILE *fp) ``` 它将字符 `c` 推回到文件 `fp` 上,若成功则返回 `c`,出错则返回 `EOF`。每个文件保证只能回推一个字符。`ungetc` 可与任何输入函数(如 `scanf`、`getc` 或 `getchar`)一起使用。 #### 1.
corwn 最低0.47元/天 解锁专栏
赠100次下载
继续阅读 点击查看下一篇
profit 400次 会员资源下载次数
profit 300万+ 优质博客文章
profit 1000万+ 优质下载资源
profit 1000万+ 优质文库回答
复制全文

相关推荐

SW_孙维

开发技术专家
知名科技公司工程师,开发技术领域拥有丰富的工作经验和专业知识。曾负责设计和开发多个复杂的软件系统,涉及到大规模数据处理、分布式系统和高性能计算等方面。
最低0.47元/天 解锁专栏
赠100次下载
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
千万级 优质文库回答免费看

最新推荐

【水管系统水头损失环境影响分析】:评估与缓解策略,打造绿色管道系统

![柯列布鲁克-怀特](https://andrewcharlesjones.github.io/assets/empirical_bayes_gaussian_varying_replicates.png) # 摘要 水管系统中的水头损失是影响流体输送效率的关键因素,对于设计、运行和维护水输送系统至关重要。本文从理论基础出发,探讨了水头损失的概念、分类和计算方法,并分析了管道系统设计对水头损失的影响。随后,本文着重介绍了水头损失的测量技术、数据分析方法以及环境影响评估。在此基础上,提出了缓解水头损失的策略,包括管道维护、系统优化设计以及创新技术的应用。最后,通过案例研究展示了实际应用的效果

Cadence AD库管理:构建与维护高效QFN芯片封装库的终极策略

![Cadence AD库管理:构建与维护高效QFN芯片封装库的终极策略](https://media.licdn.com/dms/image/C4E12AQHv0YFgjNxJyw/article-cover_image-shrink_600_2000/0/1636636840076?e=2147483647&v=beta&t=pkNDWAF14k0z88Jl_of6Z7o6e9wmed6jYdkEpbxKfGs) # 摘要 Cadence AD库管理是电子设计自动化(EDA)中一个重要的环节,尤其在QFN芯片封装库的构建和维护方面。本文首先概述了Cadence AD库管理的基础知识,并详

海洋工程仿真:Ls-dyna应用挑战与解决方案全攻略

![海洋工程仿真:Ls-dyna应用挑战与解决方案全攻略](https://media.springernature.com/lw1200/springer-static/image/art%3A10.1007%2Fs40684-021-00331-w/MediaObjects/40684_2021_331_Fig5_HTML.png) # 摘要 本文系统介绍了海洋工程仿真基础与Ls-dyna软件的应用。首先,概述了海洋工程仿真与Ls-dyna的基础知识,随后详细阐述了Ls-dyna的仿真理论基础,包括有限元分析、材料模型、核心算法和仿真模型的建立与优化。文章还介绍了Ls-dyna的仿真实践

性能瓶颈排查:T+13.0至17.0授权测试的性能分析技巧

![性能瓶颈排查:T+13.0至17.0授权测试的性能分析技巧](https://www.endace.com/assets/images/learn/packet-capture/Packet-Capture-diagram%203.png) # 摘要 本文综合探讨了性能瓶颈排查的理论与实践,从授权测试的基础知识到高级性能优化技术进行了全面分析。首先介绍了性能瓶颈排查的理论基础和授权测试的定义、目的及在性能分析中的作用。接着,文章详细阐述了性能瓶颈排查的方法论,包括分析工具的选择、瓶颈的识别与定位,以及解决方案的规划与实施。实践案例章节深入分析了T+13.0至T+17.0期间的授权测试案例

【LabView图像处理挑战】:轮廓提取解决方案的权威分享

![【LabView图像处理挑战】:轮廓提取解决方案的权威分享](https://img-blog.csdn.net/20170211210256699?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvRmFjZUJpZ0NhdA==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center) # 摘要 LabView作为一种图形化编程语言,在图像处理和轮廓提取领域提供了一种直观的实现方式。本文首先概述了LabView在图像处理中的应用,然后详细探讨了轮廓提取的基础

兼容性无忧:Hantek6254BD与软件协同操作指南

![Hantek6254BD_光盘资料.zip](https://techexplorations.com/wp-content/uploads/2019/10/techexplorations.com_oscilloscopes_for_busy_people0009-1024x576.jpg) # 摘要 本文系统地介绍了Hantek6254BD的功能特性和软件应用,分析了其软件兼容性及协同操作的实践案例,详细讨论了硬件与软件协同优化的方法以及进阶软件应用。同时,本文还提供了故障排除和设备维护的技巧,最后通过案例研究与实战演练,深入剖析了Hantek6254BD在实际操作中的性能表现和用户

【MATLAB信号处理项目管理】:高效组织与实施分析工作的5个黄金法则

![MATLAB在振动信号处理中的应用](https://i0.hdslb.com/bfs/archive/e393ed87b10f9ae78435997437e40b0bf0326e7a.png@960w_540h_1c.webp) # 摘要 本文旨在提供对使用MATLAB进行信号处理项目管理的全面概述,涵盖了项目规划与需求分析、资源管理与团队协作、项目监控与质量保证、以及项目收尾与经验总结等方面。通过对项目生命周期的阶段划分、需求分析的重要性、资源规划、团队沟通协作、监控技术、质量管理、风险应对策略以及经验传承等关键环节的探讨,本文旨在帮助项目管理者和工程技术人员提升项目执行效率和成果质

TB67S109A与PCB设计结合:电路板布局的优化技巧

![TB67S109A与PCB设计结合:电路板布局的优化技巧](https://img-blog.csdnimg.cn/direct/8b11dc7db9c04028a63735504123b51c.png) # 摘要 本文旨在介绍TB67S109A步进电机驱动器及其在PCB布局中的重要性,并详细分析了其性能特性和应用。文中探讨了TB67S109A驱动器的功能、技术参数以及其在不同应用领域的优势。同时,还深入研究了步进电机的工作原理和驱动器的协同工作方式,以及电源和散热方面的设计要求。本文还概述了PCB布局优化的理论基础,并结合TB67S109A驱动器的具体应用场景,提出了PCB布局和布线的

ISTA-2A合规性要求:最新解读与应对策略

# 摘要 随着全球化商业活动的增加,产品包装和运输的合规性问题日益受到重视。ISTA-2A标准作为一项国际认可的测试协议,规定了产品在运输过程中的测试要求与方法,确保产品能在多种运输条件下保持完好。本文旨在概述ISTA-2A的合规性标准,对核心要求进行详细解读,并通过案例分析展示其在实际应用中的影响。同时,本文提出了一系列应对策略,包括合规性计划的制定、产品设计与测试流程的改进以及持续监控与优化措施,旨在帮助企业有效应对ISTA-2A合规性要求,提高产品在市场中的竞争力和顾客满意度。 # 关键字 ISTA-2A标准;合规性要求;测试流程;案例分析;合规性策略;企业运营影响 参考资源链接:[

【游戏自动化测试专家】:ScriptHookV测试应用与案例深入分析(测试效率提升手册)

# 摘要 本文全面介绍了ScriptHookV工具的基础使用、脚本编写入门、游戏自动化测试案例实践、进阶应用技巧、测试效率优化策略以及社区资源分享。首先,文章提供了ScriptHookV的安装指南和基础概念,随后深入探讨了脚本编写、事件驱动机制、调试与优化方法。在游戏自动化测试部分,涵盖了界面元素自动化、游戏逻辑测试、以及性能测试自动化技术。进阶应用章节讨论了多线程、高级脚本功能开发和脚本安全性的管理。优化策略章节则提出了测试用例管理、持续集成流程和数据驱动测试的有效方法。最后,本文分享了ScriptHookV社区资源、学习材料和解决技术问题的途径,为ScriptHookV用户提供了一个全面的