C++linux高并发服务器项目实践 day3

news2025/1/16 18:54:47

C++linux高并发服务器项目实践 day3

  • 文件IO
    • 标准C库IO函数与LinuxIO函数
    • 虚拟地址空间
    • 文件描述符
    • Linux系统IO函数
      • open与close
    • mode:八进制的数,表示用户对创建出的新的文件的操作权限 最终的权限是:mode & ~umask 0777 r(读) w(写) x(可执行)都有这样的权限即都是1,换算成8进制就是7 三个7分别代表当前用户,当前用户所在组的权限,其他组的权限 可以通过umask来查询本机的umask的值是多少 我这里查到的是0002取反后为0775 0777 -> 11111111 & 0775 -> 11111101
      • read和write
      • lseek函数
      • stat与lstat

文件IO

内存===文件
从文件到内存为I
从内存到文件为O

标准C库IO函数与LinuxIO函数

标准C库IO函数
在这里插入图片描述

man 3 fopen
可以打开linux的操作手册关于fopen的相关资料

同样的

man 3 fwrite

linux自带的读写,使用一次就会读一次或写一次磁盘,效率其实不如c标准库的高

在这里插入图片描述

在vscode中,写入FILE,ctrl加左键可以进入他的定义,发现他的原型是_IO_FILE,再次ctrl加左键,发现该结构体中,有char*类型的各个变量用于读写和缓冲

虚拟地址空间

在这里插入图片描述
只有实际内存空间会出现的问题:

  1. 当剩余内存不足以跑下一个程序时,运行会失败
  2. 当前面的程序释放后,虽然剩余空间能跑下一个程序,但是内存是分割开的

利用MMU(内存管理单元)来进行虚拟内存和物理内存之间的管理和转换

文件描述符

在Linux的内核区,有一个PCB(进程控制块),可以控制文件描述符
文件描述符存储在文件描述符表内,文件描述符表在数据结构上是数组形式,表的默认大小为1024
在这里插入图片描述
文件描述符表的前三位是固定的
就是标准输入、标准输出、标准错误,默认是打开状态
/dev/tty 通过上述3位可以找到当前绑定的设备终端
他们对应的终端是同一个
一个文件能被打开多次,但是多次打开的文件描述符是不一样的
在fopen打开一个文件时,一个文件描述符被占用,只有将其fclose或以linux内部的close关闭后,这个文件描述符才能再次启用
内核会自己寻找空闲的最小的文件描述符

Linux系统IO函数

int open(const char *pathname,int flags);
int open(const char *pathname,int flags,mode_t mode);
int close(int fd);
ssize_t read(int fd,void *buf,size_t count);
ssize_t write(int fd,const void *buf,size_t count);
off_t lseek(int fd,off_t offset,int whence);
int stat(const char *pathname.struct stat *statbuf);
int lstat(const char *pathname,struct stat *statbuf);

open与close

虽然函数重载是c++的内容,但是上述函数是C语言的,所以两个open其实是采用可变参数的方式实现的

open函数的声明在fcntl.h这个头文件内,open内部的flags定义了一些宏,放到另外两个头文件中

第一个open函数用于打开一个已经存在的文件
int open(const char *pathname,int flags);
参数:

  • pathname:要打开的文件路径
  • flags:对文件的操作权限及其他设置
    O_RDONLY,O_WRONLY,O_RDWR 这三个设置是互斥的

返回值:

  • 返回一个新的文件描述符,如果调用失败,返回-1

errno:属于Liux系统函数库,库里面的一个全局变量,记录的是最近的错误号

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

    //打开一个已经存在的文件
    int open(const char *pathname, int flags);
        参数:
            - pathname:要打开的文件路径
            - flags:对文件的操作权限及其他设置
                O_RDONLY,O_WRONLY,O_RDWR  这三个设置是互斥的
        返回值:返回一个新的文件描述符,如果调用失败,返回-1
    errno:属于Liux系统函数库,库里面的一个全局变量,记录的是最近的错误号

    #include <stdio.h>
    void perror(const char *s);
        s参数:用户描述,比如hello,最终输出的内容是 hello:xxx(实际错误描述)

    作用:打印errno对应的错误描述

    //创建一个新的文件
    int open(const char *pathname, int flags, mode_t mode);

    #include <unistd.h>

    int close(int fd);


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

int main(){

    int fd = open("a.txt",O_RDONLY);

    if(fd == -1){
        perror("open");
    }

    //关闭
    close(fd);

    return 0;
}

第二个open函数用于创建一个新的文件
int open(const char *pathname,int flags,mode_t mode);
参数:

  • pathname:要打开的文件路径

  • flags:对文件的操作权限及其他设置
    - 必选项:O_RDONLY,O_WRONLY,O_RDWR
    - 可选项:O_CREAT 文件不存在创建文件

  • mode:八进制的数,表示用户对创建出的新的文件的操作权限
    最终的权限是:mode & ~umask
    0777 r(读) w(写) x(可执行)都有这样的权限即都是1,换算成8进制就是7
    三个7分别代表当前用户,当前用户所在组的权限,其他组的权限
    可以通过umask来查询本机的umask的值是多少
    我这里查到的是0002取反后为0775
    0777 -> 11111111
    & 0775 -> 11111101

            11111101
    

    按位与:0和任何数都为0
    umask的作用就是抹去某些权限

    可以用umask 后面跟上数值来修改例如 umask 022
    但是这个修改只能作用在本窗口,新开一个窗口就会失效

read和write

read:
头文件: #include <unistd.h>

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

  • fd:文件描述符,open得到的,通过这个文件描述符操作某个文件
  • buf:需要读取数据存放的缓冲区,数组的地址(传出参数)
  • count:指定的数组的大小

返回值:

  • 成功:
    》0:返回实际的读取到的字节数
    =0:文件已经读取完了
  • 失败: -1,并且设置errno

write:
头文件: #include <unistd.h>

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

  • fd:文件描述符,open得到的,通过这个文件描述符操作某个文件
  • buf:要往磁盘写入的数据,数组
  • count:要写的数据的实际的大小

返回值:

  • 成功:实际写入的字节数
  • 失败:返回-1,并且设置errno
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main()
{
    //1.通过open打开english.txt文件
    int srcfd = open("english.txt",O_RDONLY);
    if(srcfd == -1){
        perror("open");
        return -1;
    }

    //2.创建一个新的文件(拷贝文件)
    int destfd = open("cpy.txt",O_WRONLY|O_CREAT,0664);
    if(destfd == -1){
        perror("open");
        return -1;
    }

    //3.频繁的读写操作
    char buf[1024] = {0};

    int len = 0;
    while(( len = read(srcfd,buf,sizeof(buf))) > 0){
        write(destfd,buf,len);
    }
    

    //4.关闭文件
    close(destfd);
    close(srcfd);

    return 0;
}

lseek函数

标准C库的函数
#include <stdio.h>
int fseek(FILE *stream, long offset, int whence);

Linux系统函数
#include <sys/types.h>
#include <unistd.h>
off_t lseek(int fd, off_t offset, int whence);
第一个头文件放了一些宏定义,第二个头文件是unix的标准头文件

参数:

  • fd:文件描述符,通过open得到的,通过这个fd操作某个文件
  • offset:偏移量
  • whence:
    SEEK_SET:设置文件指针的偏移量
    SEEK_CUR:设置偏移量:当前位置+第二个参数offset的值
    SEEK_END:设置偏移量:文件大小+第二个参数offset的值

作用:

  1. 移动文件指针到头文件
    lseek(fd,0,SEEK_SET);

  2. 获取当前文件指针的位置
    lseek(fd,0,SEEK_CUR);

  3. 获取文件长度
    lseek(fd,0,SEEK_END);

  4. 拓展文件的长度,当前文件10b,110b,增加了100个字节
    lseek(fd,100,SEEK_END);
    注意:需要写一次数据

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

int main(){
    int fd = open("hello.txt",O_RDWR);

    if(fd == -1){
        perror("open");
        return -1;
    }

    //扩展文件的长度
    int ret = lseek(fd,100,SEEK_END);
    if(ret == -1)
    {
        perror("lseek");
        return -1;
    }

    //写入一个空数据
    write(fd," ",1);

    //关闭文件
    close(fd);

    return 0;
}

stat与lstat

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

int stat(const char *pathname, struct stat *statbuf);

作用:获取一个文件相关的一些信息
参数:

  • pathname:操作的文件的路径
  • statbuf:结构体变量,传出参数,用于保存获取到的文件的信息

返回值:

  • 成功:返回0
  • 失败:返回-1,设置errno

int lstat(const char *pathname, struct stat *statbuf);

作用:获取软链接文件本身的信息
参数:

  • pathname:操作的文件的路径
  • statbuf:结构体变量,传出参数,用于保存获取到的文件的信息
    返回值:
  • 成功:返回0
  • 失败:返回-1,设置errno
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<stdio.h>

int main(){
    struct stat statbuf;
    
    int ret = stat("a.txt",&statbuf);

    if(ret == -1){
        perror("stat");
        return -1;
    }

    printf("size:%ld\n",statbuf.st_size);

    return 0;
}

在这里插入图片描述
在这里插入图片描述

stat a.txt

可以查看a的一些信息

ln -s a.txt b.txt

创建一个软链接文件b.txt指向a.txt
若这里使用stat访问b.txt,则实际访问的是a.txt的信息
若想要访问b.txt本身,则需要lstat

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

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

相关文章

2021遥感应用组二等奖:近20年黄河流域植被动态与生态环境效应

作品介绍 1 研究目的 基于MODIS NDVI植被指数、土地利用数据和气象数据集&#xff0c;辅以趋势分析、偏相关分析、马尔科夫转移矩阵变化分析、多元回归分析等方法&#xff0c;全面分析黄河流域2001-2020年植被时空变化特征&#xff0c;并通过构建统计模型方式&#xff0c;定量…

【Seata】Seata配置上传Nacos

前言 在seata1.4.2版本之后&#xff0c;Seata可以通过dataId配置seata的所有配置项&#xff0c;也就是说&#xff0c;我们可以将之前上传的所有配置项整合到一个配置文件中&#xff0c;接下来直接演示该怎样操作。 如果你不清楚seata的config.txt文件在哪里下载或者其它的一些…

c++11 标准模板(STL)(std::queue)(五)

定义于头文件 <queue> template< class T, class Container std::deque<T> > class queue; std::queue 类是容器适配器&#xff0c;它给予程序员队列的功能——尤其是 FIFO &#xff08;先进先出&#xff09;数据结构。 类模板表现为底层容器的包…

kali相关操作

目录 kali换源&#xff1a; 由于没有公钥&#xff0c;无法验证下列签名&#xff1a; NO_PUBKEY 871920D1991BC93C 安装中文 设置中文输入法 kali换源&#xff1a; sudo su 备份相关配置 cp /etc/apt/sources.list /etc/apt/sources.list.bak vim /etc/apt/sources.list替…

搭建SFTP服务安全共享文件,实现在外远程访问「内网穿透」

文章目录 1.前言2.本地SFTP服务器搭建2.1.SFTP软件的下载和安装2.2.配置SFTP站点2.3.Cpolar下载和安装 3.SFTP服务器的发布3.1.Cpolar云端设置3.2.Cpolar本地设置 4.公网访问测试5.结语 1.前言 现在的网络发达&#xff0c;个人电脑容量快速上升&#xff0c;想要保存的数据资料…

最近,我们做了一次“实景”容灾演练

保障头部新闻客户端的业务连续性&#xff0c;阿里云帮助客户在真实场景下完成容灾演练。 云计算和新闻 APP&#xff0c;能有什么关系&#xff1f; 2021 年&#xff0c;传媒行业某头部媒体的新闻客户端进行了全新改版并升级上线&#xff0c;以 “内容技术”的融合驱动效率提升&a…

Vue2-黑马(十)

目录&#xff1a; &#xff08;1&#xff09;vuex-actions &#xff08;2&#xff09;vuex-调用actions &#xff08;3&#xff09;vue2实战-项目搭建 &#xff08;1&#xff09;vuex-actions 有这么一个需求&#xff0c;绿色的组件从服务器获取数据放入store&#xff0c;主…

CHAPTER 2: 《BACK-OF-THE-ENVELOPE ESTIMATION》 第2章 《初略的估计》

CHAPTER 2: BACK-OF-THE-ENVELOPE ESTIMATION 在系统设计面试中&#xff0c;有时您会被要求估计系统容量或使用粗略估计的性能需求。根据杰夫迪恩的说法&#xff0c;谷歌高级研究员&#xff0c;“粗略的计算是你使用结合思想实验和常见的性能数字&#xff0c;以获得良好的感觉…

武器目标分配问题研究进展: 模型、算法与应用

源自&#xff1a;系统公正与电子技术 作者&#xff1a;李梦杰 常雪凝 石建迈 陈超 黄金才 刘忠 摘 要 武器目标分配问题是指挥控制与任务规划领域的关键难点之一, 也是军事运筹领域的基础研究课题。经过多年研究, 武器目标分配问题在陆海空天电等领域都得到了广泛研究,…

ESLint插件开发

ESLint 插件 ESLint 插件是一个可以包含一系列ESLint 规则、配置、处理器、环境的npm模块。 创建插件 开发ESLint插件可以使用Yeoman提供的生成器来生成ESLint插件的基本项目的目录结构。 安装yeoman 和ESLint 规则的生成器 npm i -g yo generator-eslint创建eslint-xxxx使用y…

春秋云境:CVE-2022-26201(二次注入漏洞)

目录 一、题目 二、上传权限马 三、蚁剑马连接 一、题目 进入题目&#xff1a; 这个是和春秋云境&#xff1a;CVE-2022-28060一个网站&#xff0c;看来应该是存在多个漏洞。 点击admin 不用登录 Users ---- add users 选择文件 这里要是jpg图片格式 如果是php格式是无法运行…

机器学习(一)-K近邻算法(KNN)原理剖析及python源码

本篇介绍第一个机器学习算法&#xff1a;k-近邻算法&#xff0c;它非常有效而且易于掌握。首先&#xff0c;我们将探讨k-近邻算法&#xff08;KNN&#xff09;的基本理论&#xff0c;以及如何使用距离测量的方法分类物品&#xff1b;其次我们将使用Python从文本文件中导入并解析…

JavaSE学习进阶day06_02 Set集合和Set接口

第二章 Set系列集合和Set接口 Set集合概述&#xff1a;前面学习了Collection集合下的List集合&#xff0c;现在继续学习它的另一个分支&#xff0c;Set集合。 set系列集合的特点&#xff1a; Set接口&#xff1a; java.util.Set接口和java.util.List接口一样&#xff0c;同样…

D. Marcin and Training Camp(思维 + 判断一个数二进制位是否是另一个数的子集)

Problem - D - Codeforces 马辛是他大学里的一名教练。有N个学生想参加训练营。马辛是个聪明的教练&#xff0c;所以他只想派那些能冷静合作的学生参加。 让我们关注一下这些学生。每个学生可以用两个整数ai和bi来描述&#xff1b;bi等于第i个学生的技能水平&#xff08;越高越…

【刷题之路】LeetCode 程序员面试金典 08.03. 魔术索引

【刷题之路】LeetCode 程序员面试金典 08.03. 魔术索引 一、题目描述二、解题1、方法1——暴力法1.1、思路分析1.2、代码实现 2、方法2——二分分治2.1、思路分析2.2、代码实现 一、题目描述 原题连接&#xff1a; 面试题 08.03. Magic Index LCCI 题目描述&#xff1a; 魔术索…

计算机网络考试复习——第三章 3.3

3.3 使用广播信道的数据链路层 3.3.1局域网的数据链路层 局域网最主要的特点&#xff1a; 1.网络为一个单位所拥有&#xff1b; 2.地理范围和站点数目均有限。 局域网具有如下主要优点&#xff1a; 1.具有广播功能&#xff0c;从一个站点可很方便地访问全网。 2.便于系统…

分布式问题,你知道几个?

你好&#xff0c;我是田哥 现在的单体服务是很难应付面试了&#xff0c;必须要把分布式相关技术给讲清楚&#xff0c;否则面试难搞。 下面我们来聊聊&#xff0c;分布式环境下会面临哪些问题。 先来看一下主要内容&#xff1a; 分布式系统中常见的难题包括&#xff1a; 一致性问…

PackageMS 启动

1.PackageMS 相关框架类 2.PackageMS 启动过程 2.1 # SystemServer.java /** 291 * The main entry point from zygote. 292 */ 293 public static void main(String[] args) { 294 new SystemServer().run(); 295 } 上面是SystemServer的主函数。 pr…

RK3568开发板 buildroot编译配置

启扬智能RK3568开发板提供两种文件系统&#xff0c;一种是Debian&#xff0c;另外一种是BuildRoot。在使用过程中&#xff0c;有些客户需要添加属于自己的软件包&#xff0c;所以在此分享启扬RK3568关于BuildRoot编译配置的方法以及相关注意事项。 启扬提供的源码是完整sdk&am…