【Linux从入门到精通】文件I/O操作(C语言vs系统调用)

news2024/11/23 21:48:27

文章目录

一、C语言的文件IO相关函数操作

1、1 fopen与fclose

1、2 fwrite

1、3 fprintf与fscanf

1、4 fgets与fputs

二、系统调用相关接口

2、1 open与close

2、2 write和read

三、简易模拟实现cat指令

四、总结


🙋‍♂️ 作者:@Ggggggtm 🙋‍♂️

👀 专栏:Linux从入门到精通  👀

💥 标题:文件操作💥

 ❣️ 寄语:与其忙着诉苦,不如低头赶路,奋路前行,终将遇到一番好风景 ❣️ 

  本篇文章主要会讲解C语言的文件IO相关操作的函数,同时也会对Linux下的文件操作系统调用接口进行讲解。希望本篇文章会对你有所帮助。

一、C语言的文件IO相关函数操作

1、1 fopen与fclose

  fopen() 函数原型:FILE *fopen(const char *filename, const char *mode); 作用:打开一个文件,并返回一个文件指针。 参数:

  • filename:要打开的文件名(含路径)
  • mode:打开文件的模式,如 "r" 表示只读,"w" 表示写入(如果文件存在则清空内容),"a" 表示追加写入等。

  具体如下图:

  fclose() 函数原型:int fclose(FILE *stream); 作用:关闭一个打开的文件。 参数:

  • stream:要关闭的文件指针

   具体可结合下图理解:

  我们知道使用 fopen 时需要添加索要打开文件的路径。当我们以w的方式进行打开时,该文件不存在会自动创建文件,具体代码如下图:

#include<stdio.h>    
    
int main()    
{    
    FILE* fd = fopen("log.txt","w");    
    if(fd==NULL)    
    {    
        perror("fopen");    
    }    
    
    fclose(fd);  
    
    //while死循环完全是为了方便查看和观察
    while(1)
    {
        sleep(1);
    }                                                                                                                                        
    return 0;    
} 

  上述代码中,fopen中并没有添加路径,只有一个文件名字。那么能打开成功吗?其次是,当前目录下并没有 log.txt 文件,如果能打开成功,文件会被创建到哪里呢?我们带着这些疑问接着往下看。

  我们不妨先观察一下运行结果,如下图:

  我们看到确实能够出创建出来,也是创建在了当前目录了!这是为什么呢?当一个程序运行起来后,会在内存中创建相应的数据结构,同时变成进程。该进程包含了当前所在的工作目录,且还有当前的可执行文件所在的目录。具体如下图:

  所以即使我们并没有添加路径,操作系统也会知道当前所在的路径。并且默认创建到当前的工作路径下。

1、2 fwrite

  fwrite的函数原型:size_t fwrite(const void *ptr, size_t size, size_t count, FILE *stream)。用于将数据块按字节写入文件。参数:

  • ptr:指向要写入的数据的指针。
  • size:要写入的每个数据项的字节数。
  • count:要写入的数据项的数量。
  • stream:目标文件的指针。

  我们也可看下图理解:

  我们现在用fwrite向log.txt中写入,具体代码如下:

#include<stdio.h>    
#include<string.h>

int main()
{
    FILE* fp = fopen("log.txt","w");
    if(fp==NULL)
    {
        perror("fopen");
    }
   //进行文件操作
   const char *s1 = "hello Linux\n"; 
   fwrite(s1, strlen(s1), 1, fp);

  
    fclose(fp);    
    return 0;
}

   这里有一个问题:在写入文件时,要不要把s1的‘\0’写入呢?答案是不用。字符串结尾标志'\0'只是C语言规定的。文件并不用遵守C语言的规则!我们看运行结果:

  log.txt文件中确实被写入了。假如我们注释掉写入的代码,只是打开后直接关闭文件,结果会是什么呢?如下图:

  文件内容为空了!!!为什么呢?原因是以“w”的方式打开文件,就是在写入前会被清空文件内容。 

1、3 fprintf与fscanf

  fprintf() 函数原型:int fprintf(FILE *stream, const char *format, ...); 作用:向文件中按指定格式写入数据。 参数:

  • stream:要写入的文件指针。
  • format:格式化字符串,类似于printf()函数的格式化参数。 返回值:成功写入的字符数,出错时返回负值。

  fprintf与printf相比,fprintf第一个参数是FILE* ,其他的都一样。只不过是输出到了指定的文件上。

  fscanf() 函数原型:int fscanf(FILE *stream, const char *format, ...); 作用:从文件中按指定格式读取数据。 参数:

  • stream:要读取的文件指针。
  • format:格式化字符串,类似于scanf()函数的格式化参数。 返回值:成功读取并匹配的项目数量,出错或到达文件结尾时返回EOF。

  fscanf() 与scanf() 相比,fscanf() 第一个参数是FILE* ,其他的都一样。读取数据时,是从指定的文件上读取。

  这里我们就举例说明fsacnf,我们想要把文件的内容读取并输出,代码如下:

#include<stdio.h>    
#include<unistd.h>    
#include<string.h>    
    
int main()    
{    
    FILE* fp = fopen("log.txt","r");    
    if(fp==NULL)    
    {    
        perror("fopen");    
    }    
        
    char line[128];    
    while(fscanf(fp,"%s",line) != EOF)    
    {                                                                                                                                                        
        printf("%s\n", line);    
    }   
    return 0;
} 

  运行结果如下:

  注意,当我们要读取内容是,要修改打开文件的方式,应该以“r”的方式打开文件。否则会出现意想不到的结果!!!

1、4 fgets与fputs

  fgets() 函数原型:char *fgets(char *str, int n, FILE *stream); 作用:从文件中读取一行字符串。 参数:

  • str:要读取的字符串存放的缓冲区。
  • n:最多读取的字符数(包括换行符)。
  • stream:要读取的文件指针。 返回值:成功时返回str,失败或到达文件结尾时返回NULL。

   具体如下图:

  fputs() 函数原型:int fputs(const char *str, FILE *stream); 作用:向文件中写入一个字符串。 参数:

  • str:要写入的字符串。
  • stream:要写入的文件指针。 返回值:成功写入的字符数,出错时返回EOF。

 具体如下图:

  我们结合下述实例来理解fgets和fputs的用法,代码如下:

#include<stdio.h>    
#include<unistd.h>    
#include<string.h>    
    
int main()    
{    
    FILE* fp = fopen("log.txt","r");    
    if(fp==NULL)    
    {    
        perror("fopen");    
    }     
    char line[64];
    //fgets -> C -> s(string) -> 会自动在字符结尾添加\0
    while(fgets(line, sizeof(line), fp) != NULL)                                                                                                             
    {
        //printf("%s", line);
        fputs(line, stdout); //输出到屏幕上
    }
    fclose(fp);
    return 0;
} 

​

   运行结果如下:

二、系统调用相关接口

2、1 open与close

   open()函数:open函数用于打开文件并获取文件描述符。它接受一个文件路径和一组标志作为参数,并返回一个用于后续文件操作的文件描述符。函数原型:int open(const char *pathname, int flags, mode_t mode)。详细解释:

  • 函数说明:打开文件并获取文件描述符。
  • 参数:
    • pathname:要打开的文件路径。
    • flags:打开文件的标志,例如O_RDONLY(只读)、O_WRONLY(只写)、O_RDWR(读写)等。
    • mode:在创建新文件时使用的权限位。
  • 返回值:
    • 成功:返回一个非负整数,表示文件描述符
    • 失败:返回-1,并设置errno来指示错误。

  具体可结合下图理解:

  有很多的选项,我们想要添加那个选项,只需要在第二个参数 按位与(‘|’) 上就行。为什么 按位与(‘|’)呢?这里涉及到了位图的知识。想要知道的可以去了解一下位图。

  close()函数:close函数用于关闭打开的文件。它接受文件描述符作为参数,并返回一个表示成功与否的状态值。   函数原型:int close(int fd)。详细解释:

  • 函数说明:关闭打开的文件。
  • 参数:
    • fd:要关闭的文件描述符。
  • 返回值:
    • 成功:返回0。
    • 失败:返回-1,并设置errno来指示错误。

  我们通过如下实例来理解open和close。代码如下:

#include<stdio.h>    
#include<unistd.h>    
#include<string.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>

    
int main()    
{    
      int fd1 = open("log.txt", O_RDONLY);
      int fd2 = open("log2.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666); //rw-rw-rw-
      if(fd1 < 0 || fd2 < 0)
      {
          perror("open");                                                                                                                                  
          return 1;
      }
      close(fd1);
      close(fd2);

      return 0;
} 

  运行结果如下:

  确实打开成功了。为什么log2.txt与我们设置的权限并不相同呢?不要忘记了系统中还有umask掩码。

2、2 write和read

  write()函数:write函数用于向文件中写入数据。它接受文件描述符、要写入的数据和字节数作为参数,并返回实际写入的字节数。函数原型:ssize_t write(int fd, const void *buf, size_t count)。详细解释:

  • 函数说明:向文件中写入数据。
  • 参数:
    • fd:要写入的文件描述符。
    • buf:要写入的数据的缓冲区。
    • count:要写入的字节数。
  • 返回值:
    • 成功:返回实际写入的字节数。
    • 失败:返回-1,并设置errno来指示错误。

  read()函数:read函数从文件中读取数据。它接受文件描述符、缓冲区指针和要读取的字节数作为参数,并返回实际读取的字节数。函数原型:ssize_t read(int fd, void *buf, size_t count)。详细解释:

  • 函数说明:从文件中读取数据。
  • 参数:
    • fd:要读取的文件描述符。
    • buf:用于存储读取数据的缓冲区。
    • count:要读取的最大字节数。
  • 返回值:
    • 成功:返回实际读取的字节数。
    • 失败:返回-1,并设置errno来指示错误。

  write和read用起来也相对简单。这里就不再举例详细解释说明了。 

三、简易模拟实现cat指令

  我们知道,cat是打印出一个文件的内容。我们学习了文件操作后,就来简单的模拟实现一下cat指令。

  cat指令不就是接受到文件,然后打印出文件的内容吗。这好像就是我们刚刚学了文件操作。我们直接看代码:

#include<stdio.h>    
#include<unistd.h>    
#include<string.h>
int main(int argc,char* argv[])
{    
    if(argc != 2)     
    {    
        printf("argv error!\n");    
        return 1;    
    }    
    FILE *fp = fopen(argv[1], "r");     
    if(fp == NULL)    
    {    
        //strerror    
        perror("fopen");    
        return 2;    
    }    
    
    //按行读取                                                                                                                                               
    char line[64];    
    //fgets -> C -> s(string) -> 会自动在字符结尾添加\0 
    // 将文件的内容打印到屏幕上   
    while(fgets(line, sizeof(line), fp) != NULL)    
    {    
        //printf("%s", line);    
        fprintf(stdout, "%s", line); //fprintf->stdout?
    }
    fclose(fp);
    return 0;
}

  在运行时加上文件名就可以打印出文件的内容了。再加上我们之前讲到的创建子进程进行程序替换实现的简易版的shell,不就是实现了cat指令嘛!!!我们看输出结果:

四、总结

  本篇文章讲述了一系列的文件操作函数。其实我们学的C语言的文件操作函数,底层都是封装的系统调用的接口。因为我们对文件的写入和读取,不就是对硬盘的写入和读取吗!文件可是放在硬盘上的。语言想要访问硬件设备,必须通过操作系统!!! 

  每套语言都是有自己的文件操作函数,底层都是封装的系统调用的接口。但是操作系统不只是有Linux,还有windows等等。那语言就是封装所有的操作系统的接口呗。只不过是在调用时会有选择判断。这样封装后,语言就有了跨平台性。

  上文中有一个名词:文件描述符。我们并没有对此进行详解。下篇文章会对此进行讲解。这个也是一个重点!!!

  本片文章的讲解就到这里。感谢阅读ovo~ 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/872104.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JAVA多线程和并发基础面试问答(翻译)

JAVA多线程和并发基础面试问答(翻译) java多线程面试问题 1. 进程和线程之间有什么不同&#xff1f; 一个进程是一个独立(self contained)的运行环境&#xff0c;它可以被看作一个程序或者一个应用。而线程是在进程中执行的一个任务。Java运行环境是一个包含了不同的类和程序…

Shell编程之条件测试、if语句、case语句

条件语句 一、条件测试1.1 测试命令1.1 文件测试1.2 整数比较1.3 字符串比较1.4 逻辑测试1.4.1 逻辑与 &&1.4.2 逻辑或 || 1.4.3 组合应用1.5 多个命令组合执行 ( ) { } 二、if语句2.1单分支结构2.2 多分支结构2.4 if语句练习2.4.1 单分支2.4.2 简单的交互式分数反馈 三…

Shell编程之正则表达式(非常详细)

正则表达式 1.通配符和正则表达式的区别2.基本正则表达式2.1 元字符 &#xff08;字符匹配)2.2 表示匹配次数2.4 位置锚定2.5 分组 和 或者 3.扩展正则表达式4.部分文本处理工具4.1 tr 命令4.2 cut命令4.3 sort命令4.4 uniq命令 1.通配符和正则表达式的区别 通配符一般用于文件…

部署Springboot项目注意事项

步骤 步骤 1&#xff1a;将数据库内容在云服务器上的数据库部署一份 我使用mariadb&#xff1b;会出现一些不兼容现象&#xff1b;我们需要把默认值删掉 2&#xff1a;配置文件你得修改地方 a&#xff1a;linux是磁盘区分(像我自己项目用来储存验证码的文件我们得换这个配置;…

DoIP诊断入门

简介 DoIP&#xff08;Diagnosis over Internet Protocol&#xff09;是一种用于车辆诊断的网络通信协议。它基于现代互联网技术&#xff0c;允许通过以太网或IP网络进行车辆诊断和通信。 DoIP的背景是现代车辆中使用的电子控制单元&#xff08;ECU&#xff09;数量不断增加&…

利用OpenSSL实现私有 CA 搭建和证书颁发

利用OpenSSL实现私有 CA 搭建和证书颁发 一、私有 CA 搭建1. 安装openssl2. 配置 openssl3. 生成 CA 自己的私钥4. 生成 CA 自己的自签证书5. 验证自签证书 二、向私有CA申请证书流程1. 生成应用私钥文件2. 根据应用私钥生成证书申请文件3. 向CA请求颁发证书4. 验证应用证书5. …

PS/LR2024专用智能磨皮插件Portraiture提高P图效率

Portraiture 4智能磨皮插件支持Photoshop和Lightroom&#xff01;Portraiture是一款智能磨皮插件&#xff0c;为Photoshop和Lightroom添加一键磨皮美化功能&#xff0c;快速对照片中皮肤、头发、眉毛等部位进行美化&#xff0c;无需手动调整&#xff0c;大大提高P图效率。全新4…

Wlan——无线服务集和AP的基本概念以及AP的配置

目录 WLAN服务集的基本概念 AP的基本概念 AP的分类 AP模式的切换 胖&#xff08;FAT&#xff09;AP介绍 胖AP的工作模式 接入模式和路由模式的区别 胖AP的组网方式 瘦&#xff08;FIT&#xff09;AP介绍 瘦AP的工作模式 瘦AP的组网方式 胖AP和瘦AP的区别 AP的配置…

(leecode)错误的集合

最近听到的&#xff0c;还可以&#xff0c;试试吧~ 题目&#xff1a; 示例&#xff1a; 提示&#xff1a; 题解&#xff1a; 思路&#xff1a; 将数字大小的位置&#xff0c;然后遍历每个位置&#xff0c;大小为0的是缺失数字&#xff0c;大小为2的是重复数字 int* findErro…

2022年12月 C/C++(一级)真题解析#中国电子学会#全国青少年软件编程等级考试

第1题&#xff1a;加一 输入一个整数x&#xff0c;输出这个整数加1后的值&#xff0c;即x1的值。 时间限制&#xff1a;1000 内存限制&#xff1a;65536 输入 一个整数x(0 ≤ x ≤ 1000)。 输出 按题目要求输出一个整数。 样例输入 9 样例输出 10 以下是使用C语言编写的解决方案…

湘大 XTU OJ:1406 String Game、1098 素数个数 题解(非常详细)

1406 String Game 一、链接 1406 String Game 二、题目 题目描述 Alice和Bob正在玩一个基于字符串的游戏&#xff0c;一开始&#xff0c;Alice和Bob分别拥有一个等长的字符串S1和S2&#xff0c;且这两个字符串只包含小写字母。 在每个回合中&#xff0c;Alice和Bob必须分…

【Vue-Router】路由入门

路由&#xff08;Routing&#xff09;是指确定网站或应用程序中特定页面的方式。在Web开发中&#xff0c;路由用于根据URL的不同部分来确定应用程序中应该显示哪个内容。 构建前端项目 npm init vuelatest //或者 npm init vitelatest安装依赖和路由 npm install npm instal…

Python Opencv实践 - 图像旋转

import cv2 as cv import numpy as np import matplotlib.pyplot as pltimg cv.imread("../SampleImages/pomeranian.png", cv.IMREAD_COLOR)#图像旋转 #Opencv中的旋转&#xff0c;首先通过cv.getRotationMatrix2D获得旋转矩阵 #cv.getRotationMatrix2D(center,ang…

抖音小程序实现less语言编译样式

1.在抖音开发工具中搜索扩展less 2. 然后点击小齿轮选择扩展设置 3. 然后在扩展设置中选择在settings.json中编辑# 4. 在settings.json中加入以下这段代码即可 // Easy LESS配置"less.compile": {"compress": false,//是否压缩"sourceMap": fal…

腾讯云轻量应用服务器Typecho应用模板搭建博客流程

腾讯云百科分享使用腾讯云轻量应用服务器Typecho应用模板搭建博客流程&#xff0c;Typecho 是开源的博客建站平台&#xff0c;具有轻量、高效、稳定等特点&#xff0c;操作界面简洁友好。该镜像基于 CentOS 7.6 64 位操作系统&#xff0c;并已预置 Nginx、PHP、MariaDB 软件。您…

学点Selenium玩点新鲜~,让分布式测试有更多玩法

前 言 我们都知道 Selenium 是一款在 Web 应用测试领域使用的自动化测试工具&#xff0c;而 Selenium Grid 是 Selenium 中的一大组件&#xff0c;通过它能够实现分布式测试&#xff0c;能够帮助团队简单快速在不同的环境中测试他们的 Web 应用。 分布式执行测试其实并不是一…

Unity2D RPG开发笔记 P1 - Unity界面基础操作和知识

文章目录 工具选择简单快捷键Game 窗口分辨率检视器Transform 组件Sprite Renderer综合检视器 工具选择 按下 QWERTY 可以选择不同的工具进行 旋转、定位、缩放 简单快捷键 按下 Ctrl D 可以复制物体 Game 窗口分辨率 16:9 为最常见的分辨率 检视器 Transform 组件 物体在…

django-基本环境配置

文章目录 django 环境安装1. 安装环境1.1 安装 Python (配置虚拟环境)1.1.1 步骤 1.2 Conda配置环境参考 django 环境安装 1. 安装环境 1.1 安装 Python (配置虚拟环境) 由于国外源速度慢&#xff0c;可以pip添加清华源 pip config set global.index-url https://pypi.tuna.…

什么是CSS的box-sizing属性?它有哪些取值,各有什么不同?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ CSS的box-sizing属性⭐ 取值⭐ 不同之处⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启航前端之旅 欢迎来到前端入门之旅&#xff01;这个专栏是为那些对Web…

python单元测试框架(测试固件、批量执行)

python测试框架 在Python语言中应用最广泛的单元测试框架是unittest和pytest,unittest属于标准库&#xff0c;只要安装了Python解释器后就可以直接导入使用了,pytest是第三方的库&#xff0c;需要单独的安装。 1.白盒测试原理 在软件架构的层面来说&#xff0c;测试最核心的…