问题:同一个进程中,先后对同一个文件描述符进行写入 / 读取 操作,读写指针的位置导致读取内容失败

本文主要是介绍问题:同一个进程中,先后对同一个文件描述符进行写入 / 读取 操作,读写指针的位置导致读取内容失败,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文件‘读写指针’ / lseek 系统调用
  1. 在进行文件操作(读 / 写)时,可以使用 lseek 系统调用来设置文件的下一个读写位置。
  2. 多进程 / 多线程 环境中,对同一资源进行读 / 写操作时,很可能会 因为 ‘读写指针’ 的变化导致各种隐蔽的问题,这一点必须要仔细。
  3. lseek 系统调用(对文件描述符的读写指针进行设置,即可以用它来 设置文件的下一个读写位置
  4. open 函数的 O_APPEND 选项,会影响文件的 ‘读写指针’ 是否会在多个进程间共享(是:导致写操作 追加到文件末尾,否:导致写操作 相互覆盖)(尝试解决一个问题的最好方式是:立刻开始动手操作)
#include <unistd.h>
#include <sys/types.h>
off_t lseek(int fildes, off_t offset, int whence);

实验代码如下:

#include <stdio.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdlib.h>void main(){// 以读写方式打开文件,如果文件不存在则新建,新建文件时文件权限为 ‘rw-r--r--’int fd = open("1111111111111.txt", O_RDWR | O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);if(fd < 0){printf("error: open failed.\n");exit(EXIT_FAILURE);}// 先执行写入操作int res = write(fd, "abc", 3);if(res != 3){printf("error: write failed.\n");exit(EXIT_FAILURE);}char buf[10];// 尝试读取之前写入的内容res = read(fd, buf, 3);if(res > 0){buf[res] = '\0';printf("readed:%s\n", buf);}else if(res == 0){printf("EOF\n");}else{printf("error: read failed.\n");exit(EXIT_FAILURE);}
}

运行结果为:

  • 程序创建新文件后,立即执行写入操作,随后从中读取之前写入的数据
  • 问题是: read 调用返回 0,表示已经到达文件末尾。但实际上文件中是有数据的
ubuntu@cuname:~/dev/beginning-linux-programming/test$ gcc -o test4 test4.c
ubuntu@cuname:~/dev/beginning-linux-programming/test$ ./test4
EOF 	// 读取失败(到达文件末尾)
ubuntu@cuname:~/dev/beginning-linux-programming/test$ ls -l | grep 1111
-rw-r--r-- 1 ubuntu ubuntu    3 Apr  8 17:23 1111111111111.txt
ubuntu@cuname:~/dev/beginning-linux-programming/test$ cat 1111111111111.txt
abc

!!!突然想到可能是 ‘读写指针’ 的问题!!!
(***因为执行写入操作后,‘读写指针’ 始终指向文件的末尾!***)
修改程序代码如下(在写入成功后,设置 ‘读写指针’ 的位置):

#include <stdio.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdlib.h>void main(){int fd = open("1111111111111.txt", O_RDWR | O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);if(fd < 0){printf("error: open failed.\n");exit(EXIT_FAILURE);}int res = write(fd, "abc", 3);if(res != 3){printf("error: write failed.\n");exit(EXIT_FAILURE);}// 设置文件的 ‘读写指针’ 的位置,// 将 ‘读写指针’ 相对于当前位置(即:文件末尾)向左偏移 res 个字节(即:回到指执行写入操作时的位置)。off_t off = lseek(fd, 0 - res, SEEK_CUR);if(off == -1){printf("error: lseek failed.\n");exit(EXIT_FAILURE);}char buf[10];res = read(fd, buf, 3);if(res > 0){buf[res] = '\0';printf("readed:%s\n", buf);}else if(res == 0){printf("EOF\n");}else{printf("error: read failed.\n");exit(EXIT_FAILURE);}
}

运行程序,结果如下(确实是文件 ’读写指针‘ 的问题!):

ubuntu@cuname:~/dev/beginning-linux-programming/test$ gcc -o test4 test4.c
ubuntu@cuname:~/dev/beginning-linux-programming/test$ ./test4
readed:abc 	// 输出正确

这篇关于问题:同一个进程中,先后对同一个文件描述符进行写入 / 读取 操作,读写指针的位置导致读取内容失败的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/586877

相关文章

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

Python将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

使用Jackson进行JSON生成与解析的新手指南

《使用Jackson进行JSON生成与解析的新手指南》这篇文章主要为大家详细介绍了如何使用Jackson进行JSON生成与解析处理,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 核心依赖2. 基础用法2.1 对象转 jsON(序列化)2.2 JSON 转对象(反序列化)3.

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

C#使用SQLite进行大数据量高效处理的代码示例

《C#使用SQLite进行大数据量高效处理的代码示例》在软件开发中,高效处理大数据量是一个常见且具有挑战性的任务,SQLite因其零配置、嵌入式、跨平台的特性,成为许多开发者的首选数据库,本文将深入探... 目录前言准备工作数据实体核心技术批量插入:从乌龟到猎豹的蜕变分页查询:加载百万数据异步处理:拒绝界面

Python使用自带的base64库进行base64编码和解码

《Python使用自带的base64库进行base64编码和解码》在Python中,处理数据的编码和解码是数据传输和存储中非常普遍的需求,其中,Base64是一种常用的编码方案,本文我将详细介绍如何使... 目录引言使用python的base64库进行编码和解码编码函数解码函数Base64编码的应用场景注意

Mysql表的简单操作(基本技能)

《Mysql表的简单操作(基本技能)》在数据库中,表的操作主要包括表的创建、查看、修改、删除等,了解如何操作这些表是数据库管理和开发的基本技能,本文给大家介绍Mysql表的简单操作,感兴趣的朋友一起看... 目录3.1 创建表 3.2 查看表结构3.3 修改表3.4 实践案例:修改表在数据库中,表的操作主要

C# WinForms存储过程操作数据库的实例讲解

《C#WinForms存储过程操作数据库的实例讲解》:本文主要介绍C#WinForms存储过程操作数据库的实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、存储过程基础二、C# 调用流程1. 数据库连接配置2. 执行存储过程(增删改)3. 查询数据三、事务处

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip