[Linux打怪升级之路]-文件操作

简介: [Linux打怪升级之路]-文件操作

一、认识操纵系统下的文件

1、什么是文件

在平常我们对于文件的认识都是在window操纵系统下认识,我们建立的目录是文件,文本文档是文件,图片是文件。

在Linux中有这么一句话:Linux下一切皆文件

为什么这么说的呢?

Linux中所有的内容都是以文件的形式保存的,我们认为普通文件是文件,一个目录我们也称为文件,甚至认为硬件设备(键盘,硬盘,打印机等)是文件。

Linux有一个根目录,其他的所以的文件都放在根目录中。

2、文件的类型

普通文件

直接就可以使用的我们就称为普通文件,如上面的makefile就是一个普通文件

目录文件

这个目录中包含各个文件的文件名和文件及其指向这些文件的指针,只要有权限就可以访问目录中的任何文件,上面的myshell就是一个目录文件,一个目录不仅仅有目录名,还有一些权限设置、文件大小等。

其他的一些文件类型:字符设备文件和块设文件、套接字文件(socket)、符号链接文件(symbolic link)、管道文件(pipe)。

3、文件的共识

  1. 空文件,也要在磁盘中占据空间。
  2. 文件 = 内容 +属性。
  3. 文件操纵 = 对内容 + 对属性.
  4. 如果没有指明对应的文件的路径,和默认当前路径。
  5. 当我们通过fopen,fclose,fwrite,fread对文件进行操纵的时候,编译代码,形成可执行的程序,但是不运行,对于的文件操纵执行了吗?没有对文件的操纵,本质上是进程对文件的操纵
  6. 一个文件没有被打开,可以直接进行文件的访问吗?不能

通过上面的共识我们可以得出,对文件的操纵的本质 :是进程和打开文件的关系

二、系统级的文件操作接口

我们在C语言和C++都有对文件的操纵函数,我们都知道文件是存放在磁盘上的,而要想访问文件就必须将磁盘上的文件导入到内存中,在进行相应的操作,其实本质上是操作系统在对这些文件进行管理,而C语言和C++中对文件的操作的接口,他底层也是操作系统对文件操作的接口,只是通过了封装了而已。

1、文件打开open/文件关闭close

文件打开open

头文件:<sys/types.h>、 <sys/stat.h>、<fcntl.h>

访问形式:

1、int open(const char *pathname, int flags);

2、int open(const char *pathname, int flags, mode_t mode);

当我们要调用open接口就必须包含相应的头文件,但这里要注意的是这里接口为我们提供了二个调用的方式,方式1是文件已经存在我们调用方式2是文件不存在时要调用的接口

接口参数:

const char *pathname:这里就指我们要打开的文件名

int flags:指的是标记位,传过来的比特位,不同的比特位就调用不同选项,从对文件进行一些初始化。

mode_t mode:这里是设置相应的权限(如操作系统默认的普通文件的权限是0x666)

这里我们要重点了解open接口的第二个参数,上面我们说他是一个标记位 ,我们知道一个int是有32个比特位了,这里我们用每个比特位表示不同的选项。

下面我们看一段代码理解一下:

#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <assert.h>
 
// 每一个宏,对应的数值,只有一个比特位是1,彼此位置不重叠
#define ONE (1<<0)
#define TWO (1<<1)
#define THREE  (1<<2)
#define FOUR (1<<3)
 
void show(int flags)
{
    if(flags & ONE) printf("one\n");
    if(flags & TWO) printf("two\n");
    if(flags & THREE) printf("three\n");
    if(flags & FOUR) printf("four\n");
}
 
int main()
{
    show(ONE);
    printf("-----------------------\n");
    show(TWO);
    printf("-----------------------\n");
    show(ONE | TWO);
    printf("-----------------------\n");
    show(ONE | TWO | THREE);
    printf("-----------------------\n");
    show(ONE | TWO | THREE | FOUR);
    printf("-----------------------\n");
    return 0;
}

当我们运行代码,就会发现我们通过不同的比特位就调用了不同的参数选项,展现出现不同的效果。

flags多个选项:

  • O_RDONLY: 只读打开
  • O_WRONLY: 只写打开
  • O_RDWR : 读,写打开
  • 这三个常量,必须指定一个且只能指定一个
  • O_CREAT : 若文件不存在,则创建它。需要使用mode选项,来指明新文件的访问权
  • O_APPEND: 追加写
  • O_TRUNC如果文件已经存在,且成功打开,则删除文件中原来的全部数据

对于这些选项我们还可以进行功能的多选,选项间用 “ | ” 分割

O_RDONY | O_CREAT//这里的文件用只读的方式创建,如果没有这个文件会自动创建

返回值

  • 成功:新打开的文件描述符(fd)
  • 失败:-1

文件关闭 close

文件关闭接口用起来非常简单,只要给他传打开文件的文件描述符就可以了。

2、其他的文件接口

read读文件

函数原型:

ssize_t read(int fd, void *buf, size_t count);

参数

  • fd:是文描述符
  • buf:接收读取数据的缓存区
  • count:读取的字节数
  • 返回值:读取成功则返回读取的字节数,读取到文件尾则返回0,读取失败则返回-1,同时设置全局变量errno的值来表示错误原因;

write写文件

函数原型:

ssize_t write(int fd, const void *buf, size_t count);

参数:

  • fd:是文描述符
  • buf:存放写入数据的缓存区
  • count:写入的的字节数
  • 返回值:写入成功则返回实际写入的字节数,写入失败则返回-1,同时设置全局变量errno的值来表示错误原因;

3、代码演示文件操作

下面为了更好的理解文件操作,为了大家演示了文件的打开,写入和读取数据,最后在关闭。

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
 
int main()
{
    //定义文件名
    char filename[] = "pjb.txt";
    //定义写入的数据
    char write_data[] = "hellow Linux!";
    //读取的数据
    char read_buf[64] = { 0 };
 
    //测试写入数据
    int fd = open(filename, O_RDWR | O_CREAT, O664);
    if (fd < 0)
    {
        perror("open");
        return -1;
    }
    //写入内容
    int ret = write(fd, write_data, sizeof(write_data));
    if (ret < 0)
    {
        perror("write");
        return -1;
    }
    else
    {
        printf("write: %s\n", write_data);
    }
    //关闭文件
    close(fd);
    //测试写入文件
    fd = open(filename, O_RDONLY);;
    if (fd < 0)
    {
        perror("open");
        return -1;
    }
    //写入
    ret = read(fd, read_buf, sizeof(read_buf));
    if (ret < 0)
    {
        perror("read");
        return -1;
    }
    else
    {
        printf("read: %s\n", read_buf);
    }
    close(fd);
    return 0;
}

这里观察到我们成功向文件中写入和向文件中读取数据。


相关文章
|
8月前
|
安全 Linux 编译器
BigCloud Enterprise Linux 8和Rocky Linux 8升级OpenSSH步骤
本文介绍了在BigCloud Enterprise Linux 8.2/8.6和Rocky Linux 8.10上升级OpenSSH的详细步骤。首先配置Telnet服务和GCC编译器以确保远程登录安全,接着备份旧版OpenSSH并下载、编译、安装最新版本(如9.9p1)。然后创建新的sshd系统服务配置文件,调整配置并启动服务。最后验证升级效果,关闭不必要的Telnet服务,并处理可能的防火墙和SELinux问题。通过这些步骤可有效修复低版本OpenSSH带来的高危漏洞。
500 13
|
9月前
|
应用服务中间件 Linux nginx
【Azure App Service】基于Linux创建的App Service是否可以主动升级内置的Nginx版本呢?
基于Linux创建的App Service是否可以主动升级内置的Nginx版本呢?Web App Linux 默认使用的 Nginx 版本是由平台预定义的,无法更改这个版本。
266 77
|
安全 Linux 网络安全
Linux端的ssh如何升级?
Linux端的ssh如何升级?
1046 59
|
运维 监控 网络协议
运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面
本文介绍了运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面,旨在帮助读者提高工作效率。从基本的文件查看与编辑,到高级的网络配置与安全管理,这些命令是运维工作中的必备工具。
919 3
|
人工智能 监控 Shell
常用的 55 个 Linux Shell 脚本(包括基础案例、文件操作、实用工具、图形化、sed、gawk)
这篇文章提供了55个常用的Linux Shell脚本实例,涵盖基础案例、文件操作、实用工具、图形化界面及sed、gawk的使用。
2397 2
|
Linux TensorFlow 算法框架/工具
在Linux上安装其他版本的cmake 或 升级cmake
在Linux上安装其他版本的cmake 或 升级cmake
1405 2
|
Ubuntu Linux 数据库
在Linux中,如何进行软件包升级?
在Linux中,如何进行软件包升级?
|
Linux 数据安全/隐私保护
在Linux中,如何进行文件系统的迁移和升级?
在Linux中,如何进行文件系统的迁移和升级?