【Linux】应用编程之C语言文件操作

简介: 【Linux】应用编程之C语言文件操作

前言

  对于文件IO来说,一切都是围绕文件描述符来进行的,在Linux系统中所有打开的文件都是一个对应的文件描述符。
  文件描述符的本质是一个非负整数,当我们打开一个文件时,系统会给我们分配一个文件描述符。
  当我们对一个文件做读写操作的时候,我们使用open函数的时候返回的就是这个文件的文件描述符,我们后续对文件的操作例如read和write都需要使用这个参数。

完成目标:

  在windows上编程和Linux上的文件操作有所不同(如果需要学习windows上的文件操作编程可以跳转到这里:C/C++文件读取操作),本文主要是熟悉Linux上的文件操作的接口使用和相关文档的翻译。


一、工程的创建

1、目录结构

2、Makefile

object = *.o
CC = gcc
main:$(object)
        $(CC) $(object) -o main
        rm *.o
        file main
%.o:%.c
        $(CC) -c $<
clean:
        rm *.o
        rm main

这里复制需要注意一下需要将复制过去的空格替换成TAB

二、文件的操作

1.创建、打开、关闭文件:open、close、creat

如需要查看函数的完整说明使用以下命令查看详细的说明;

man open 
man close  

下面是打开文件的函数接口以及使用该接口需要包含的头文件

// 使用系列操作需要包含得头文件
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <fcntl.h>
// 文件操作的接口函数:我们常用的是open
 int open(const char *pathname, int flags); // 如果没有创建文件得权限使用该打开方式
 int open(const char *pathname, int flags, mode_t mode);// 如果打开文件发现没有得情况需要创建选项这里就需要填写mode权限

const char* pathname传入的是文件名的字符串地址

flags表示文件的打开方式,如下表

参数 含义
O_CREAT 必要时创建文件
O_TRUNC 删除全部现有数据
O_APPEND 维持现有数据,保存到其后面
O_RDONLY 只读打开
O_WRONLY 只写打开
O_RDWR 读写打开

第二个函数还提供一个mode_t的参数该参数是表示创建文件的权限设置,这里类似chmod指令,在commod下使用ls -al 或则 ll指令可以查看文件权限

-rw-rw-r-- 1 stylle stylle  291 May 24 10:36 file.c
-rwxrwxr-x 1 stylle stylle 8384 May 24 10:16 main
-rw-rw-r-- 1 stylle stylle  129 May 24 10:03 Makefile

r表示可读 -4

w表示可写 - 2

x表示可执行 -1

所以我们常用到得chmod 777 就表示将该文件赋予可读可写可执行得权限;我们在操作完文件过后一定要释放文件,所以我们open过后一定要跟close函数,就如同malloc和free一样;

#include <unistd.h>
int close(int fd); // 直接传入我们刚刚open函数返回的文件描述符

完整的使用用例

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char **argv)
{
        int fd;
        fd = open("file_tmp.c", O_CREAT | O_RDWR,0666);
        if(fd < 0)
        {
                perror("open file error\r\n");
        }
        else{
                printf("fd = %d\r\n",fd);
                close(fd);
        }
        return 0;
}

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int creat(const char *pathname, mode_t mode);

pathname为创建文件的名称

mode表示创建文件的权限

返回的参数为创建的文件描述符,如果小于0表示文件创建失败,下面是程序演示

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
int main(int argc, char **argv)
{
  int fd;
  if(argc < 2)
  {
    perror("plese input creat filename\r\n");
    exit(0);
  }
  int ret = creat(argv[1], 0666);
  if(ret < 0)
  {
    perror("creat file error\r\n");
    exit(0);
  }
  printf("creat file %s success\r\n",argv[1]);
  return 0;
}

2.读取、写入文件:read、write

然后我们开始向文件中写入内容然后从中读取出来,这里我们需要使用到write和read,然后我们查看函数需要传入的参数

#include <unistd.h>
ssize_t write(int fd, const void *buf, size_t count);
ssize_t read(int fd, void *buf, size_t count);

write

fd表示文件描述符,这里填写我们open函数返回的fd就可以了

buf表示要写入文件的字符串

count表示要写入字符的个数

read

fd表示文件描述符,这里填写我们open函数返回的fd就可以了

buf表示接收读取到的字符串

count表示要读取的字符个数

然后我们来实验一下

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
int main(int argc, char **argv)
{
  int fd;
  char read_str[15];
  int ret = 0;
  fd = open("tmp.txt", O_CREAT | O_RDWR,0666);
  if(fd < 0)
  {
    perror("open file error\r\n");
    exit(0);
  }
  ret = write(fd, "hello world", strlen("hello world")+1);// +1是strlen未包含‘\0’的长度
  if(ret < 0)
  {
    perror("write error\r\n");
  }
  close(fd);
  fd = open("tmp.txt", O_RDONLY);
  if(fd < 0)
  {
    perror("open read only file error\r\n");
    exit(0);
  }
  ret = read(fd, read_str, ret);
  close(fd);
  if(ret < 0)
  {
    perror("read error\r\n");
  }
  else
  {
    printf("read_num = %d read:%s\r\n",ret,read_str); 
  }
  return 0;
}

3.文件定位:lseek

下面是打开文件的函数接口以及使用该接口需要包含的头文件

#include <sys/types.h>
#include <unistd.h>
off_t lseek(int fd, off_t offset, int whence);

fd 文件描述符

offset 文件指针位移位置如果为正整数表示向后位移如果为负整数表示向前位移

whence 一共有三个参数SEEK_SET、SEEK_CUR、SEEK_END分别表示从文件头、当前位置、文件尾开始位移

返回值为当前位移指针的位置,如果位移出错返回-1,下面来个小实验

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, char **argv)
{
  int fd;
  int ret;
  char read_str[10];
  fd = open("a.txt", O_RDONLY);
  if(fd < 0)
  {
    perror("open error");
    exit(0);
  }
  // a.txt: "hello stylle"
  ret = read(fd, read_str, 5);// read "hello"
  if(ret < 0)
  {
    perror("read error");
  }
  else
  {
    ret = lseek(fd, 0, SEEK_CUR);// return cur file addr
    read_str[5] = '\0';
    printf("read %d:%s\n",ret, read_str);
    
    ret = lseek(fd, 0, SEEK_END);
    printf("file size:%d\n",ret);
    lseek(fd, -7, SEEK_CUR);// '\0'
    memset(read_str, 0, 10);
    ret = read(fd, read_str, 6);
    if(ret > 0)
    {
      printf("read 6 digits:%s\n",read_str);
    }
    close(fd);
  }
  return 0;
}

最后

本章需要注意的点:

1、写入文件后不能马上读取需要重新打开读取

2、上述的内容并没有十分完整,完整函数的用法和原型可以使用man 函数名称 进行查阅


相关文章
|
3月前
|
存储 程序员 C语言
c语言文件操作
本文介绍了C语言中文件操作的核心概念,包括文件缓冲区、数据与程序文件的区别、流的概念及标准输入输出流。重点讲解了文件的打开与关闭、读写操作(如fgetc、fputs、fscanf、fwrite等),以及随机读写函数fseek、ftell和rewind的使用方法。同时说明了文本文件与二进制文件的读取结束判定方式,并通过代码示例展示了实际应用。
|
6月前
|
安全 应用服务中间件 网络安全
在Linux环境部署Flask应用并启用SSL/TLS安全协议
至此,你的Flask应用应该能够通过安全的HTTPS协议提供服务了。记得定期更新SSL证书,Certbot可以帮你自动更新证书。可以设定cronjob以实现这一点。
422 10
|
9月前
|
C语言
C语言中条件操作符的应用
最后,条件操作符是个超级英雄,但不是每个代码问题都需要一个超级英雄来解决。一定要在适当的时候适度的使用它,那么它将成为你的编程工具箱中的一件强力工具。
440 75
|
6月前
|
Java Linux 网络安全
Linux云端服务器上部署Spring Boot应用的教程。
此流程涉及Linux命令行操作、系统服务管理及网络安全知识,需要管理员权限以进行配置和服务管理。务必在一个测试环境中验证所有步骤,确保一切配置正确无误后,再将应用部署到生产环境中。也可以使用如Ansible、Chef等配置管理工具来自动化部署过程,提升效率和可靠性。
675 13
|
5月前
|
存储 数据采集 监控
ubuntu(linux)系统主要应用于哪些工业场景中?研维三防ubuntu系统的手持工业三防平板电脑在哪些行业中有实际应用
Ubuntu 系统凭借其独特的优势,在众多工业场景中得到了广泛应用,为工业数字化、智能化发展提供了有力支持。而研维三防基于 Ubuntu 定制系统的手持工业三防平板电脑,更是将 Ubuntu 系统的优势与工业级的性能、坚固耐用性完美结合,在电力、物流、制造等多个行业中展现出强大的应用价值,助力企业提高生产效率、优化管理流程、提升产品质量,成为推动工业现代化发展的重要力量。随着技术的不断进步与创新,相信 Ubuntu 系统以及研维三防这类工业级设备将在更多的工业领域中发挥更大的作用,为工业发展带来更多的机遇与变革。
|
8月前
|
安全 算法 Ubuntu
Linux(openssl)环境:编程控制让证书自签的技巧。
总结:在Linux环境中,OpenSSL是一个非常实用的工具,可以帮助我们轻松地生成自签名证书。通过上述三个简单步骤,即可为内部网络、测试环境或开发环境创建自签名证书。但在公共访问场景下,建议购买经过权威认证机构签发的证书,以避免安全警告。
386 13
|
7月前
|
存储 编译器 程序员
c语言的文件操作与文件缓冲区
如果没有文件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。磁盘(硬盘)上的文件是文件。但是在程序设计中,我们⼀般谈的⽂件有两种:程序文件、数据文件(从文件功能的角度来分类 的)。就比如说我们电脑中以.txt为后缀的就是文件的一种,他就是数据文件。.exe为后缀的就为程序文件。函数名功能适用范围fgetc字符输入函数所有输入流fputc字符输出函数所有输出流fgets。
192 0
|
10月前
|
JavaScript Ubuntu Linux
如何在阿里云的linux上搭建Node.js编程环境?
本指南介绍如何在阿里云Linux服务器(Ubuntu/CentOS)上搭建Node.js环境,包含两种安装方式:包管理器快速安装和NVM多版本管理。同时覆盖全局npm工具配置、应用部署示例(如Express服务)、PM2持久化运行、阿里云安全组设置及外部访问验证等步骤,助你完成开发与生产环境的搭建。
|
11月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
通过本文,您可以了解如何在业务线程中注册和处理Linux信号。正确处理信号可以提高程序的健壮性和稳定性。希望这些内容能帮助您更好地理解和应用Linux信号处理机制。
224 26
|
11月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。
226 17