• 内存映射文件详解-----C++实现


    先不说内存映射文件是什么。贴个代码先,。

    #include <iostream>
    #include <fcntl.h>
    #include <io.h>
    #include <afxwin.h>
    using namespace std;
    
    int main()
    {
        //开始
        //获得文件句柄
        HANDLE hFile=CreateFile(
            "c:\test.dat",   //文件名
            GENERIC_READ|GENERIC_WRITE, //对文件进行读写操作
            FILE_SHARE_READ|FILE_SHARE_WRITE,
            NULL,     
            OPEN_EXISTING,  //打开已存在文件
            FILE_ATTRIBUTE_NORMAL,   
            0);  
    
        //返回值size_high,size_low分别表示文件大小的高32位/低32位
        DWORD size_low,size_high;
        size_low= GetFileSize(hFile,&size_high); 
    
        //创建文件的内存映射文件。   
        HANDLE hMapFile=CreateFileMapping(  
            hFile,     
            NULL,   
            PAGE_READWRITE,  //对映射文件进行读写
            size_high,    
            size_low,   //这两个参数共64位,所以支持的最大文件长度为16EB
            NULL);   
        if(hMapFile==INVALID_HANDLE_VALUE)   
        {   
            AfxMessageBox("Can't create file mapping.Error%d:
    ",   GetLastError());   
            CloseHandle(hFile);
            return 0;   
        }  
    
        //把文件数据映射到进程的地址空间
        void* pvFile=MapViewOfFile(
            hMapFile, 
            FILE_MAP_READ|FILE_MAP_WRITE, 
            0,
            0,
            0);  
        unsigned char *p=(unsigned char*)pvFile; 
    
        //至此,就获得了外部文件test.dat在内存地址空间的映射,
        //下面就可以用指针p"折磨"这个文件了
        CString s;
        p[size_low-1]='!'; 
        p[size_low-2]='X'; //修改该文件的最后两个字节(文件大小<4GB高32位为0)
        s.Format("%s",p);
        //读文件的最后3个字节
        AfxMessageBox(s);
    
    
        //结束
        //UnmapViewOfFile(pvFile); //撤销映射
        //CloseHandle(hFile); //关闭文件
    
        return 0;
    }


    忘小了说,只要你把这几个API函数搞定了,一般的内存映射问题就可以解决了。。但是内存映射文件到底是干嘛的呢?让我们先来思考一个

    如果您想读的内容大于系统分配的内存块怎么办?如果您想搜索的字符串刚好超过内存块的边界又该如何处理?对于第一个问题,您也许会说,只要不断地读就不解决了吗。至于第二个问题,您又会说在内存块的边界处做一些特别的处理,譬如放上一些标志位就可以了。原理上确实是行得通,但是这随问题复杂程度加深而显得非常难以处理。其中的第二个问题是有名的边界判断问题,程序中许许多多的错误都是由此引起。想一想,如果我们能够分配一个能够容纳整个文件的大内存块该多好啊,这样这两个问题不都迎刃而解了吗?是的,WIN32的内存映射文件确实允许我们分配一个装得下现实中可能存在的足够大的文件的内存。

    利用内存映射文件您可以认为操作系统已经为您把文件全部装入了内存,然后您只要移动文件指针进行读写即可了。这样您甚至不需要调用那些分配、释放内存块和文件输入/输出的API函数,另外您可以把这用作不同的进程之间共享数据的一种办法。运用内存映射文件实际上没有涉及实际的文件操作,它更象为每个进程保留一个看得见的内存空间。至于把内存映射文件当成进程间共享数据的办法来用,则要加倍小心,因为您不得不处理数据的同步问题,否则您的应用程序也许很可能得到过时或错误的数据甚至崩溃。本课中我们将主要讲述内存映射文件,将不涉及进程间的同步。WIN32中的内存映射文件应用非常广泛,譬如:即使是系统的核心模块---PE格式文件装载器也用到了内存映射文件,因为PE格式的文件并不是一次性加载到内存中来的,譬如他它在首次加载时只加载必需加载的部分,而其他部分在用到时再加载,这正好可以利用到内存映射文件的长处。实际中的大多数文件存取都和PE加载器类似,所以您在处理该类问题时也应该充分利用内存映射文件。

    内存映射文件本身还是有一些局限性的,譬如一旦您生成了一个内存映射文件,那么您在那个会话期间是不能够改变它的大小的。所以内存映射文件对于只读文件和不会影响其大小的文件操作是非常有用的。当然这并不意味着对于会引起改变其大小的文件操作就一定不能用内存影射文件的方法,您可以事先估计操作后的文件的可能大小,然后生成这么大小一块的内存映射文件,然后文件的长度就可以增长到这么一个大小。

  • 相关阅读:
    Redis5设计与源码分析 (第17章 HyperLogLog相关命令的实现)
    Redis5设计与源码分析 (第16章 GEO相关命令)
    ES5和ES6函数的this指向
    vue响应式原理 (响应式并不等于数据双向绑定,千万不要混淆)
    vue中data为什么是函数而不是对象
    vue-enum 前端常量 枚举工具
    Vue3 写业务逻辑不适合用TS(TypeScript)
    vue-property-decorator vue3 ts 用的装饰器
    github git clone下载加速 && npm install 下载加速
    vue3 如果用ts,导出时候要用 defineComponent,这俩是配对的,为了类型的审查正确
  • 原文地址:https://www.cnblogs.com/keanuyaoo/p/3265312.html
Copyright © 2020-2023  润新知