【Linux 内核 内存管理】内存管理系统调用 ④ ( 代码示例 | mmap 创建内存映射 | munmap 删除内存映射 )

举报
韩曙亮 发表于 2022/04/14 01:19:42 2022/04/14
【摘要】 文章目录 一、mmap 创建内存映射代码示例1、fopen 打开或创建文件2、lseek 设置文件大小3、mmap 函数使用4、munmap 删除内存映射 二、完整代码示例 ...





一、mmap 创建内存映射代码示例




1、fopen 打开或创建文件


使用 fopen 函数 , 打开一个文件 , 此时文件可能不存在 , 需要创建文件 ;

    // 打开文件
    fd = open(argv[1], O_CREAT | O_RDWR | O_TRUNC, 00777);

2、lseek 设置文件大小


通过 lseek 函数 , 设置文件的大小 , 将文件偏移 sizeof(student) * 10 - 1 大小 , 就是设置文件大小设置为 10 个 student 结构体大小

    // 修改文件偏移量 , 将文件的读写位置指向文件头后 , 
    // 再增加 sizeof(student) * 10 - 1 偏移量 , 偏移量从 0 开始计算 ,
    // 该操作的作用是将文件大小设置为 10 个 student 结构体大小
    lseek(fd, sizeof(student) * 10 - 1, SEEK_SET);

3、mmap 函数使用


调用 mmap 函数 , 创建文件映射 , 相关参数作用如下 :

  • NULL : 映射区的开始地址
  • sizeof(student) * 1 : 文件映射区的长度
  • PROT_READ | PROT_WRITE : 内存保护的标志位 , 该内存页的内容可以 读取 写入
  • MAP_SHARED : 指定映射关系 , 指的是该映射是进程的共享内存空间
  • fd : 文件描述符 , 被映射的文件
  • 0 : 被映射文件的偏移量 , 从文件的哪个字节位置开始映射

如果返回 -1 指针 , 则说明 内存映射 创建失败 ;

    // 创建文件映射
    // NULL : 映射区的开始地址
    // sizeof(student) * 1 : 文件映射区的长度
    // PROT_READ | PROT_WRITE : 内存保护的标志位 , 该内存页的内容可以 读取 写入 
    // MAP_SHARED : 指定映射关系 , 指的是该映射是进程的共享内存空间
    // fd : 文件描述符 , 被映射的文件 
    // 0 : 被映射文件的偏移量 , 从文件的哪个字节位置开始映射
    p_student = (student*)mmap(NULL, sizeof(student) * 10, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

    // mmap 文件映射创建失败
    if (p_student == (void*) - 1)
    {
        printf("mmap 文件映射创建失败 !");
        return -1;
    }

    // 创建完文件映射之后 , 文件描述符就可以释放了
    close(fd);

4、munmap 删除内存映射


调用 munmap 函数 , 删除 mmap 创建的 内存映射 ;

    // 删除文件映射 
    munmap(p_student, sizeof(student) * 10);




二、完整代码示例



#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <fcntl.h>

#include <sys/mman.h>
#include <sys/types.h>


/*
    定义一个结构体 代表 " 学生 "
    结构体成员中设置一个 char* 字符串 和 int 类型数据
    分别代表 学生的 姓名 和 年龄
 */
typedef struct
{
    char name[4];  // 姓名
    int age;        // 年龄
}student;


int main(int argc, char** argv)
{
    // 打开文件的 文件描述符
    int fd;

    // 循环控制变量
    int i;

    // 学生结构体指针 , 指向 student 结构体类型变量
    student* p_student;

    // 用于生成姓名字符串
    char name_char;

    // 打开文件
    fd = open(argv[1], O_CREAT | O_RDWR | O_TRUNC, 00777);

    // 修改文件偏移量 , 将文件的读写位置指向文件头后 , 
    // 再增加 sizeof(student) * 10 - 1 偏移量 , 偏移量从 0 开始计算 ,
    // 该操作的作用是将文件大小设置为 10 个 student 结构体大小
    lseek(fd, sizeof(student) * 10 - 1, SEEK_SET);

    // 向文件中写入数据 , 生成文件 
    write(fd, "", 1);

    // 创建文件映射
    // NULL : 映射区的开始地址
    // sizeof(student) * 1 : 文件映射区的长度
    // PROT_READ | PROT_WRITE : 内存保护的标志位 , 该内存页的内容可以 读取 写入 
    // MAP_SHARED : 指定映射关系 , 指的是该映射是进程的共享内存空间
    // fd : 文件描述符 , 被映射的文件 
    // 0 : 被映射文件的偏移量 , 从文件的哪个字节位置开始映射
    p_student = (student*)mmap(NULL, sizeof(student) * 10, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

    // mmap 文件映射创建失败
    if (p_student == (void*) - 1)
    {
        printf("mmap 文件映射创建失败 !");
        return -1;
    }

    // 创建完文件映射之后 , 文件描述符就可以释放了
    close(fd);


    // 逐个字节拷贝
    name_char = 'A';
    for (i = 0; i < 10; i++)
    {
        // 将字符串的第 1 个字节设置为 '\0' , 这是 字符串的结尾 , 
        // 第 0 个字节就是字符串的实际内容 , 该字符串只有 1 个字符
        (*(p_student + i)).name[1] = '\0';

        // 拷贝 字符串 到 p_student 指向的内存中 , 该内存是文件映射内存
        // 拷贝内存的同时 , 也会修改文件内容
        memcpy((*(p_student + i)).name, &name_char, 1);

        // 设置
        (*(p_student + i)).age = 1 + i;

        // 生成不同的字符 , 用于生成不同的 name 字符串
        name_char++;
    }

    printf("文件初始化完毕 !\n");

    // 休眠 8 秒
    sleep(8);

    // 删除文件映射 
    munmap(p_student, sizeof(student) * 10);

    printf("mmap 文件映射展示完毕 !\n");
    return 0;
}

编译并执行代码 : 上述源码保存在 mmap_demo_01.c 文件中 , 执行

gcc mmap_demo_01.c -o mmap_demo_01

命令 , 编译上述源码 , 并输出可执行文件 mmap_demo_01 , 执行

./mmap_demo_01 file

命令 , 开始执行该应用程序 ;

执行结果如下 :

han@ubuntu:~/vscode/mmap$ gcc mmap_demo_01.c -o mmap_demo_01
han@ubuntu:~/vscode/mmap$ ./mmap_demo_01 file
文件初始化完毕 !
mmap 文件映射展示完毕 !
han@ubuntu:~/vscode/mmap$ 

在这里插入图片描述

文章来源: hanshuliang.blog.csdn.net,作者:韩曙亮,版权归原作者所有,如需转载,请联系作者。

原文链接:hanshuliang.blog.csdn.net/article/details/124137260

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。