• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆-23  評(píng)論-73  文章-3  trackbacks-0
            內(nèi)存映射文件:

              內(nèi)存映射文件有三種,第一種是可執(zhí)行文件的映射,第二種是數(shù)據(jù)文件的映射,第三種是借助頁(yè)面交換文件的內(nèi)存映射.應(yīng)用程序本身可以使用后兩種內(nèi)存映射.

            1.可執(zhí)行文件映射:
              Windows在執(zhí)行一個(gè)Win32應(yīng)用程序時(shí)使用的是內(nèi)存映射文件技術(shù).系統(tǒng)先在進(jìn)程地址空間的0x00400000以上保留一個(gè)足夠大的虛擬地址空間(0x00400000以下是由系統(tǒng)管理的),然后把應(yīng)用程序所在的磁盤空間作為虛擬內(nèi)存提交到這個(gè)保留的地址空間中去(我的理解也就是說(shuō),虛擬內(nèi)存是由物理內(nèi)存和磁盤上的頁(yè)面文件組成的,現(xiàn)在應(yīng)用程序所在的磁盤空間就成了虛擬地址的頁(yè)面文件).做好這些準(zhǔn)備后,系統(tǒng)開始執(zhí)行這個(gè)應(yīng)用程序,由于這個(gè)應(yīng)用程序的代碼不在內(nèi)存中(在頁(yè)面文件中),所以在執(zhí)行第一條指令的時(shí)候會(huì)產(chǎn)生一個(gè)頁(yè)面錯(cuò)誤(頁(yè)面錯(cuò)誤也就是說(shuō),系統(tǒng)所訪問的數(shù)據(jù)不在內(nèi)存中),系統(tǒng)分配一塊內(nèi)存把它映射到0x00400000處,把實(shí)際的代碼或數(shù)據(jù)讀入其中(系統(tǒng)分配一塊內(nèi)存區(qū)域,把它要訪問的在頁(yè)面文件中的數(shù)據(jù)讀入到這塊內(nèi)存中,需在注意是系統(tǒng)讀入代碼或數(shù)據(jù)是一頁(yè)一頁(yè)讀入的),然后可以繼續(xù)執(zhí)行了.當(dāng)以后要訪問的數(shù)據(jù)不在內(nèi)存中時(shí),就可以通過(guò)前面的機(jī)制訪問數(shù)據(jù).對(duì)于Win32DLL的映射也是同樣,不過(guò)DLL文件應(yīng)該是被Win32進(jìn)程共享的(我想應(yīng)該被映射到x80000000以后,因?yàn)?x80000000-0xBFFFFFFF是被共享的空間).

              當(dāng)系統(tǒng)在另一個(gè)進(jìn)程中執(zhí)行這個(gè)應(yīng)用程序時(shí),系統(tǒng)知道這個(gè)程序已經(jīng)有了一個(gè)實(shí)例,程序的代碼和數(shù)據(jù)已被讀到內(nèi)存中,所以系統(tǒng)只需把這塊內(nèi)存在映射到新進(jìn)程的地址空間即可,這樣不就實(shí)現(xiàn)了在多個(gè)進(jìn)程間共享數(shù)據(jù)了嗎!然而這種共享數(shù)據(jù)只是針對(duì)只讀數(shù)據(jù),如果進(jìn)程改寫了其中的代碼和數(shù)據(jù),操作系統(tǒng)就會(huì)把修改的數(shù)據(jù)所在的頁(yè)面復(fù)制一份到改寫的進(jìn)程中(我的理解也就是說(shuō)共享的數(shù)據(jù)沒有改變,進(jìn)程改寫的數(shù)據(jù)只是共享數(shù)據(jù)的一份拷貝,其它進(jìn)程在需要共享數(shù)據(jù)時(shí)還是共享沒有改寫的數(shù)據(jù)),這樣就可以避免多個(gè)進(jìn)程之間的相互干擾.

            2.數(shù)據(jù)文件的內(nèi)存映射:
              數(shù)據(jù)文件的內(nèi)存映射原理與可執(zhí)行文件內(nèi)存映射原理一樣.先把數(shù)據(jù)文件的一部分映射到虛擬地址空間的0x80000000 - 0xBFFFFFFF,但沒有提交實(shí)際內(nèi)存(也就是說(shuō)作為頁(yè)面文件),當(dāng)有指令要存取這段內(nèi)存時(shí)同樣會(huì)產(chǎn)生頁(yè)面錯(cuò)誤異常.操作系統(tǒng)捕獲到這個(gè)異常后,分配一頁(yè)內(nèi)存,映射內(nèi)存到發(fā)生異常的位置,然后把要訪問的數(shù)據(jù)讀入到這塊內(nèi)存,繼續(xù)執(zhí)行剛才產(chǎn)生異常的指令(這里我理解的意思是把剛才產(chǎn)生異常的指令在執(zhí)行一次,這次由于數(shù)據(jù)已經(jīng)映射到內(nèi)存中,指令就可以順利執(zhí)行過(guò)去),由上面的分析可知,應(yīng)用程序訪問虛擬地址空間時(shí)由操作系統(tǒng)管理數(shù)據(jù)在讀入等內(nèi)容,應(yīng)用程序本身不需要調(diào)用文件的I/O函數(shù)(這點(diǎn)我覺得很重要,也就是為什么使用內(nèi)存映射文件技術(shù)對(duì)內(nèi)存的訪問就象是對(duì)磁盤上的文件訪問一樣).

            3.基于頁(yè)面交換文件的內(nèi)存映射:
              內(nèi)存映射的第三種情況是基于頁(yè)面交換文件的.一個(gè)Win32進(jìn)程利用內(nèi)存映射文件可以在進(jìn)程共享的地址空間保留一塊區(qū)域(0x8000000 - 0xBFFFFFFF),這塊區(qū)域與系統(tǒng)的頁(yè)面交換文件相聯(lián)系.我們可以用這塊區(qū)域來(lái)存儲(chǔ)臨時(shí)數(shù)據(jù),但更常見的做法是利用這塊區(qū)域與其他進(jìn)程通信(因?yàn)?x80000000以上是系統(tǒng)空間,進(jìn)程切換只是私有地址空間,系統(tǒng)空間是所有進(jìn)程共同使用的),這樣多進(jìn)程間就可以實(shí)現(xiàn)通信了.事實(shí)上Win32多進(jìn)程間通信都是使用的內(nèi)存映射文件技術(shù),如PostMessage(),SentMessage()函數(shù),在內(nèi)部都使用內(nèi)存映射文件技術(shù).

            使用內(nèi)存映射文件的方法:
            1.利用內(nèi)存映射文件進(jìn)行文件I/O操作:
              CreateFile()-->CreateFileMapping()-->MapViewOfFile()......

            2.利用內(nèi)存映射文件實(shí)現(xiàn)Win32進(jìn)程間通信:
            我只介紹兩種常用的方法:
              第一種方法:兩個(gè)進(jìn)程使用同一個(gè)文件映射核心對(duì)象,打開各自的視圖,或者父進(jìn)程把自己創(chuàng)建的文件映射核心對(duì)象繼承給子進(jìn)程使用.這種方法比較安全有效.
              第二種方法:基于頁(yè)面交換文件的內(nèi)存映射對(duì)象.在調(diào)用CreateFileMapping()函數(shù)時(shí),傳遞的文件句柄為0xFFFFFFFF,系統(tǒng)就從頁(yè)面交換文件中提交物理內(nèi)存,然后進(jìn)程之間按照第一種方法進(jìn)程通信.這種方法不用事先準(zhǔn)備一個(gè)特殊的文件(也就是說(shuō)不用事先調(diào)用CreateFile()返回一個(gè)文件的句柄),非常方便.

            注:轉(zhuǎn)載請(qǐng)保持文章完整和原文鏈接
            posted on 2008-04-19 13:08 ViskerWong 閱讀(6788) 評(píng)論(5)  編輯 收藏 引用

            評(píng)論:
            # re: 內(nèi)存映射文件學(xué)習(xí)筆記 2009-01-01 01:56 | iamatig
            說(shuō)得相當(dāng)好,我覺得比windows核心編程要說(shuō)得好!說(shuō)得具體清楚!而且我認(rèn)為最多細(xì)節(jié)上與實(shí)際有出入,但結(jié)果應(yīng)該是一樣的!
            正解了我的惑!感謝!!!!!!
            我花了幾天時(shí)間來(lái)理解書上說(shuō)的,感覺好難受,因?yàn)楦杏X作者對(duì)術(shù)語(yǔ)的使用有些隨意,有時(shí)會(huì)混用而不說(shuō)明限制范圍,或者說(shuō)沒很清楚的說(shuō)明前提。
            今天晚上正理解得差不多了,再搜索到貴文,感覺非常棒!非常地具體!
              回復(fù)  更多評(píng)論
              
            # re: 內(nèi)存映射文件學(xué)習(xí)筆記 2009-01-01 01:57 | iamatig
            做好這些準(zhǔn)備后,系統(tǒng)開始執(zhí)行這個(gè)應(yīng)用程序,由于這個(gè)應(yīng)用程序的代碼不在內(nèi)存中(在頁(yè)面文件中),所以在執(zhí)行第一條指令的時(shí)候會(huì)產(chǎn)生一個(gè)頁(yè)面錯(cuò)誤(頁(yè)面錯(cuò)誤也就是說(shuō),系統(tǒng)所訪問的數(shù)據(jù)不在內(nèi)存中),引起系統(tǒng)分配一塊內(nèi)存把它映射到0x00400000處,把實(shí)際的代碼或數(shù)據(jù)讀入其中(系統(tǒng)分配一塊內(nèi)存區(qū)域,把它要訪問的在頁(yè)面文件中的數(shù)據(jù)讀入到這塊內(nèi)存中,需在注意是系統(tǒng)讀入代碼或數(shù)據(jù)是一頁(yè)一頁(yè)讀入的),然后可以繼續(xù)執(zhí)行了.

            這段說(shuō)得不錯(cuò)。核心編程中也有。

            另外的就是覺得頁(yè)面文件這個(gè)術(shù)語(yǔ)作者有時(shí)侯好像只是指虛擬內(nèi)存,不包括物理內(nèi)存。  回復(fù)  更多評(píng)論
              
            # re: 內(nèi)存映射文件 學(xué)習(xí)筆記 2009-10-10 15:32 | zwz_good
            說(shuō)得好,學(xué)習(xí)了。  回復(fù)  更多評(píng)論
              
            # re: 內(nèi)存映射文件 學(xué)習(xí)筆記 2011-12-22 12:17 | jc_ontheroad
            學(xué)習(xí)了。  回復(fù)  更多評(píng)論
              
            # re: 內(nèi)存映射文件 學(xué)習(xí)筆記[未登錄] 2014-05-24 09:47 | jlz
            樓主如果這是你的原創(chuàng),我只能說(shuō),請(qǐng)確認(rèn)自己理解正確了再發(fā),不能確認(rèn)就說(shuō)明。錯(cuò)誤連篇,簡(jiǎn)直看不下去。  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            国产精品内射久久久久欢欢 | 久久久这里只有精品加勒比| 国内精品久久久久久久97牛牛| 噜噜噜色噜噜噜久久| 一日本道伊人久久综合影| 久久久久久国产精品免费免费| 狠狠久久综合| 欧美精品福利视频一区二区三区久久久精品 | 国产精品久久一区二区三区 | 久久久精品人妻一区二区三区蜜桃| 久久精品桃花综合| 中文字幕乱码久久午夜| 久久久久无码精品国产不卡| 高清免费久久午夜精品| 国产欧美一区二区久久| 国产成人精品久久亚洲| 久久综合狠狠综合久久97色| 亚洲国产天堂久久综合| 亚洲女久久久噜噜噜熟女| 国产午夜精品理论片久久影视 | 久久se精品一区二区| 一本大道加勒比久久综合| 亚洲欧洲精品成人久久奇米网| 精品熟女少妇AV免费久久| 久久人人爽人人爽人人片av高请| 99久久精品国产麻豆| 久久人人爽人爽人人爽av| 久久天天躁狠狠躁夜夜网站| 久久久久综合网久久| 色综合久久精品中文字幕首页 | www久久久天天com| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 亚洲欧美日韩精品久久亚洲区 | 色偷偷偷久久伊人大杳蕉| 久久久综合九色合综国产| 久久久午夜精品福利内容| 国产精品视频久久| 久久久国产打桩机| 91精品国产综合久久久久久| 久久精品无码专区免费| 大伊人青草狠狠久久|