Linux 系统编程 开篇/ 文件的打开/创建

news/2024/11/16 12:23:42/

从本节开始学习关于Linux系统编程的知识!

学习Linux的系统编程有非常多的知识点,在应用层面,很重要的一点就是学习如何“用代码操作文件来实现文件创建,打开,编辑等自动化执行

那如何自动化实现对文件的创建,打开,编辑等呢?答案就是使用Linux系统提供的一系列API函数(如 open, write/read, lseek,close等等)。

Linux文件编程的一般步骤

总结一下就是,打开文件,以及读写操作之后关闭文件的操作都不是必不可少的。 

Linux 文件管理 

简单来说,现在学习的用户层面的操作就是给内核发送指令,让内核来驱动物理磁盘进行操作 

 

打开/创建文件

需要包含的头文件:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

相关的API函数: 

int open(const char *pathname, int flags); //pathname是文件路径,flags是权限
int open(const char *pathname, int flags, mode_t mode);int creat(const char *pathname, mode_t mode);

open函数

当调用open打开一个文档之后,open函数会返回一个非负整数,这个整数就是文件描述符

如果对一个文档都进行了open,并返回了“2”,之后又想对其进行write,就是write(2)

如果返回为负数,则说明open失败!

也就是说,

文件标识符0可以指代键盘输入

文件标识符1可以指代键盘输出

且文件描述符只在当前进程有效!

参数说明

pathname

  • 要打开的文件名(含路径)

flags : 

  • O_RDONLY 只读打开
  • O_WRONLY 只写打开
  • O_RDWR 可读可写打开

当附带了权限后,打开的文件就只能按照这种权限来操作

以上这三个参数中应当只指定一个。下列参数是可选择的:

  • O_CREAT:若文件不存在则创建它。使用此选项时,需要同时说明第三个参数mode.用其说明该新文件的存取许可权限。
  • O_EXCL:以这种属性去打开文件时,如果同时指定了O_CREAT,而文件已经存在,则打开文件失败(也可以说返回值是-1)。
  • O_APPEND:以这种属性去打开文件时,每次写时都加到文件的尾端。(不想写入时文件被覆盖,用这个flag,正常写入没有加其他条件的话,原来文件中的数据内容会被覆盖,注意是覆盖,覆盖本身的字节长度,没有覆盖的会保留,不是删除)
  • O_TRUNC:以这种属性去打开文件时,如果这个文件中本来是有内容的, 而且为只读或只写成功打开,则将其长度截短为0,通俗点的意思就是把原来的文件中的内容删除,写入你自己要的数据内容
  • Mode:一定是在flags中使用了 O-CREAT 标志, mode 记录待创建的文件的访问权限

关于mode的值和权限的对应关系:

共有三种权限

  • 是否可读(r), 对应数字4
  • 是否可写(w), 对应数字2
  • 是否可执行(x), 对应数字1

共有三个权限分配的对象

  • 主用户
  • 同组用户
  • 其他组用户

比如:给的是0600时,则对应权限“-rw-------”,即只给主用户分配可读可写(4+2=6)的权限同组用户(第三位)和其他组用户(第四位)没有任何权限

除了0600这种表达方式,Linux也直接提供了一些宏来表示:

其中较为常用的是:

  • S_IRWXU:对主用户来说可读,可写,可执行
  • S_IRUSR:对主用户来说可读
  • S_IWUSR:对主用户来说可写
  • S_IXUSR:对主用户来说可执行

实操演示

1. 创建一个名为“FILE”的文件夹,将文件部分的学习代码都放进去,并在cmd中cd到FILE:

2. 使用touch指令创建一个新的文件:

关于touch指令,可以参考:

Linux命令之touch命令_touch linux_恒悦sunsite的博客-CSDN博客

3.  使用vi命令创建一个c文件:

关于vi模式下的一些操作,之前接触过,参考:

Linux 系统初识_mjmmm的博客-CSDN博客

4.  使用man命令查看open相关的函数:

man的使用参考:

Linux下的man命令_linux man命令_邓永豪的博客-CSDN博客

简单来说就是 “man + 函数对应的手册号 + 函数名” 其中手册号是1~9,常用前3个,实在不知道可以一个个试,手册号不加也没事

然后就可以看到对于open函数的描述,这样就方便在Linux系统里面直接查阅和复制

5. 编写demo1.c,然后保存退出:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>int main()
{       int fd; // file descriptionfd = open("./file1",O_RDWR);printf("file description = %d\n",fd);return 0;
}       

 6. 运行demo1.c:

可见,我创建的文档“file1”被自动打开,并分配了文件描述符“3”

7. 此时,如果将 file1 删除,再执行一遍代码:

可见,返回-1,返回失败

O_CREAT参数的应用

在实际应用中,我如果不希望一个文件不存在就直接返回失败,那么可以在open函数中,添加上面提到的“O_CREAT”参数,就可以“若文件不存在则创建它”,并按要求增加“mode”参数:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>int main()
{int fd; // file descriptionfd = open("./file1",O_RDWR|O_CREAT, 0600); //0600代表即将创建的文件“可读可写”printf("file description = %d\n",fd);return 0;
}

此时,在file1不存在的情况下open,依然会返回正常的文件标识符,并创建file1:

O_EXCL参数的应用

O_EXCL:以这种属性去打开文件时,如果同时指定了O_CREAT,而文件已经存在,则打开文件失败(也可以说返回值是-1)。

重写demo1.c:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>int main()
{int fd; // file descriptionfd = open("./file1",O_RDWR|O_CREAT|O_EXCL, 0600);if(fd == -1){printf("fail to open, file already exit\n");}else{printf("file description = %d\n",fd);}return 0;
}

可见,当file1存在时,会返回-1,无法打开文件;只有将file1删除,才会正常返回文件标识符。 

O_APPEND参数的应用

O_APPEND:以这种属性去打开文件时,每次写时都加到文件的尾端。(不想写入时文件被覆盖,用这个flag,正常写入没有加其他条件的话,原来文件中的数据内容会被覆盖,注意是覆盖,覆盖本身的字节长度,没有覆盖的会保留,不是删除)

关于写入的操作,在下一节有介绍!

不加O_APPEND写入时:

使用demo2.c的代码执行两边,第二遍的时候,将写入的数据从“mjmmjm”改成“123”:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>int main()
{int fd; // file descriptionchar *buf = "mjmmjm"; //第一遍char *buf = "123"; //第二遍fd = open("./file1",O_RDWR|O_CREAT, 0600);printf("file description = %d, open successfully!\n",fd);write(fd, buf, strlen(buf));close(fd); //close after writing return 0;
}

 

可见,如果没有添加O_APPEND,则第二次写入的数据会覆盖在原数据之上。

加入O_APPEND写入时:

依然使用demo2.c的代码执行两边,第二遍的时候,将写入的数据从“mjmmjm”改成“123”:

唯一的区别时这一次加上O_APPEND参数:

fd = open("./file1",O_RDWR|O_CREAT|O_APPEND, 0600);

可见,添加了O_APPEND了之后,就自动在文件末尾添加内容了。

O_TRUNC参数的应用

O_TRUNC:以这种属性去打开文件时,如果这个文件中本来是有内容的, 而且为只读或只写成功打开,则将其长度截短为0,通俗点的意思就是把原来的文件中的内容删除,写入你自己要的数据内容

在刚刚O_APPEND的实验中,file1的内容被修改成了“mjmmjm123”,现在修改demo2.c的代码,加入O_RRUNC参数,不删除file1的前提下编译运行 demo2.c:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>int main()
{int fd; // file descriptionchar *buf = "new";fd = open("./file1",O_RDWR|O_CREAT|O_TRUNC, 0600);printf("file description = %d, open successfully!\n",fd);write(fd, buf, strlen(buf));close(fd); //close after writing return 0;
}

可见,文件的内容被完全替换成了新的写入内容,原来的内容全部被删除了。

creat函数

(需要添加的库在开头就说明了)

int creat(const char *pathname, mode_t mode);

参数说明

  • pathname:要打开的文件名(含路径)
  • mode:待创建的文件的访问权限

mode值的选择见上面的说明。

  • 返回值:文件标识符

creat函数的原型等价于 open(pathname,O_CREAT | O_TRUNC | O_WRONLY,mode);

其中,O_CREAT和O_TRUNC上面已经提到过,一个负责“文件不存在就创建它”,一个负责“每次写入就将之前内容全部删除”,最后一个O_WRONLY就是open函数中的flag参数,表示“只写打开

实操演示

修改demo1.c:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>int main()
{int fd; // file descriptionfd = creat("./file2", S_IRWXU); //给主用户赋予可读可写可执行的权限if(fd == -1){printf("fail to open, file already exit\n");}else{printf("file description = %d\n",fd);}return 0;
}

可见,creat函数 在file2不存在的情况下,会自动创建,印证了creat函数天生就带有open函数中的O_CREAT功能。


http://www.ppmy.cn/news/1010165.html

相关文章

学python的心得体会1000字,学python的心得体会2000字

这篇文章主要介绍了学python的心得体会2000字&#xff0c;具有一定借鉴价值&#xff0c;需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获&#xff0c;下面让小编带着大家一起了解一下。 1. 初学者应该从简单的练习开始&#xff0c;先掌握基本的语法和概念&#xff0c;…

【网络基础实战之路】基于三个分公司的内网搭建并连接运营商的实战详解

系列文章传送门&#xff1a; 【网络基础实战之路】设计网络划分的实战详解 【网络基础实战之路】一文弄懂TCP的三次握手与四次断开 【网络基础实战之路】基于MGRE多点协议的实战详解 【网络基础实战之路】基于OSPF协议建立两个MGRE网络的实验详解 PS&#xff1a;本要求基于…

6年资深测试整理,接口测试总结,你不知道的都在这了...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 接口测试 是测试…

如何将 dubbo filter 拦截器原理运用到日志拦截器中?

业务背景 我们希望可以在使用日志拦截器时&#xff0c;定义属于自己的拦截器方法。 实现的方式有很多种&#xff0c;我们分别来看一下。 拓展阅读 java 注解结合 spring aop 实现自动输出日志 java 注解结合 spring aop 实现日志traceId唯一标识 java 注解结合 spring ao…

Java - 两种判断闰年的方法

&#x1f914;️判断一个年份是否是闰年的常规方法是遵循以下规则&#xff1a; 如果年份能够被4整除&#xff0c;但不能被100整除&#xff0c;那么它是闰年。如果年份能够被400整除&#xff0c;那么它也是闰年。 boolean b1 (y & 3) 0 && ((y % 100 ! 0) || (y %…

Netty:通过Channel发送ByteBuf的数据,只发送可读的字节

说明 使用Netty Channel发送数据的时候&#xff0c;如果发送的数据存放在ByteBuf中&#xff0c;那么只会发送ByteBuf中可读的字节。即便容量&#xff08;capacity&#xff09;大于可读字节数&#xff0c;那也不会多发送数据。 示例 代码片段 package com.thb.power.termina…

嵌入式开发学习(STC51-13-温度传感器)

内容 通过DS18B20温度传感器&#xff0c;在数码管显示检测到的温度值&#xff1b; DS18B20介绍 简介 DS18B20是由DALLAS半导体公司推出的一种的“一线总线&#xff08;单总线&#xff09;”接口的温度传感器&#xff1b; 与传统的热敏电阻等测温元件相比&#xff0c;它是一…

Gof23设计模式之组合模式

1.定义 ​组合模式又名部分整体模式&#xff0c;是用于把一组相似的对象当作一个单一的对象。组合模式依据树形结构来组合对象&#xff0c;用来表示部分以及整体层次。这种类型的设计模式属于结构型模式&#xff0c;它创建了对象组的树形结构。 2.结构 组合模式主要包含三种…