• <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>

            大龍的博客

            常用鏈接

            統(tǒng)計

            最新評論

            Windows 下的進程間通訊及數(shù)據(jù)共享

            Windows 下的進程間通訊及數(shù)據(jù)共享

            Windows 下有很多方法實現(xiàn)進程間通訊,比如用 socket,管道(Pipe),信箱(Mailslot),等等。但最基本最直接的還是使用內存共享。其他方法最終還是會繞道這里。

            可想而知,如果物理內存只有一份,讓這份內存在不同的進程中,映射到各自的虛擬地址空間上,每個進程都可以讀取同一份數(shù)據(jù),是一種最高效的數(shù)據(jù)交換方法。下面我們就討論如何實現(xiàn)它。

            共享內存在 Windows 中是用 FileMapping 實現(xiàn)的。我們可以用 CreateFileMapping 創(chuàng)建一個內存文件映射對象, CreateFileMapping 這個 API 將創(chuàng)建一個內核對象,用于映射文件到內存。這里,我們并不需要一個實際的文件,所以,就不需要調用 CreateFile 創(chuàng)建一個文件, hFile 這個參數(shù)可以填寫 INVALID_HANDLE_VALUE 。但是,文件長度是需要填的。Windows 支持長達 64bit 的文件,但是這里,我們的需求一定不會超過 4G , dwMaximumSizeHigh 一定是 0 ,長度填在 dwMaximumSizeLow 即可。然后調用 MapViewOfFile 映射到當前進程的虛擬地址上即可。一旦用完共享內存,再調用 UnmapViewOfFile 回收內存地址空間。

            Windows 把 CreateFileMapping 和 MapViewOfFile 兩個 API 分開做是有它的道理的。這是因為允許映射一個超過 4G 的文件,而地址空間最大只有 4G (實際上,一般用戶的程序只能用到 2G) , MapViewOfFile 就可以指定文件的 Offset 而只映射一部分。

            在 CreateFileMapping 的最后一個參數(shù) pszName 填寫一個名字,那么別的進程就可以用這個名字去調用 OpenFileMapping 來打開這個 FileMapping 對象,在新的進程內作映射。 不過,通過約定字符串的方法似乎不太優(yōu)雅。

            一個優(yōu)雅的方法是,用 DuplicateHandle 在新進程中復制一份 FileMapping 對象出來,然后想辦法把 Handle 通知新進程,比如用消息的方式傳遞過去。

            如果需要共享內存的兩個進程是父子關系,那么我們可以不用消息傳遞的方式來通知 FileMapping 的 Handle 。父進程可以用繼承 Handle 的方式直接把 FileMapping 的 Handle 傳遞到子進程中。當然,在 CreateFileMapping 時就應該設置可以被繼承的屬性。

            大約是這樣:

            SECURITY_ATTRIBUTES sa;
            sa.nLength=sizeof(sa);
            sa.lpSecurityDescriptor=NULL;
            sa.bInheritHandle=TRUE;
            handle=CreateFileMapping(INVALID_HANDLE_VALUE,&sa,PAGE_READWRITE,0,size,NULL);

            這樣,在 CreateProcess 的時候,如果 bInheritHandles 參數(shù)為 TRUE ,所有有可被繼承屬性的內核對象都會被復制到子進程中。

            注:內核對象的繼承就是在 CreateProcess 創(chuàng)建子進程,但是子進程的主線程尚未活動之前,內核掃描當前進程中所有內核對象,檢查出有可繼承屬性的那些,再用 DuplicateHandle 復制一份到子進程。由于是內核對象,在內核中實質只有一份,所有只是引用記數(shù)加一,父進程和子進程對同一內核對象的 Handle 一定是相同的。

            復制內核對象的過程是由 CreateProcess 內部完成的,我們可以放心的把對象 Handle (和子進程相同) 通過命令行傳遞給子進程。或者,用環(huán)境變量傳遞也可以。

            值得注意的是,子進程用完了這個 FileMapping 對象后一樣需要 CloseHandle 減去引用計數(shù)。

            備注:
            CreateProcess 調用時,pszCommandLine 不能直接填上一個不可修改的字符串。例如:

            CreateProcess("test.exe","test argument",...);

            這樣就是錯誤的,因為 "test argument" 會被編譯器編譯放到不可修改的數(shù)據(jù)段中。正確的方法是:

            char cmdline[]="test argument";
            CreateProcess("test.exe",cmdline,...);

            這樣,命令行的字符串就被放在堆棧上,是可以被讀寫的。

            CreateProcess 的倒數(shù)第二個參數(shù)需要填寫一個 STARTUPINFOW 結構,這個結構很復雜,通常填起來很麻煩。我們可以復制一份父進程的結構,再酌情修改。方法是:

            STARTUPINFO si={sizeof(si)};
            PROCESS_INFORMATION pi;
            GetStartupInfo(&si);
            CreateProcess(...,&si,& pi);

            這里, STARTUPINFO 結構的第一個長度信息通常應該填上,保證 GetStartupInfo(&si); 的正確執(zhí)行。

            posted on 2007-06-12 15:48 大龍 閱讀(1003) 評論(0)  編輯 收藏 引用

            91视频国产91久久久| 久久久无码精品亚洲日韩京东传媒| 国产精品久久久久a影院| 久久人人添人人爽添人人片牛牛| 精品国产99久久久久久麻豆| 欧美牲交A欧牲交aⅴ久久| 精品国产91久久久久久久| 久久久久国产成人精品亚洲午夜| 久久AAAA片一区二区| 性欧美丰满熟妇XXXX性久久久 | 浪潮AV色综合久久天堂| 久久综合久久久| 久久人做人爽一区二区三区| 久久国产精品成人免费| 热综合一本伊人久久精品| 精品久久久久久中文字幕人妻最新 | 久久亚洲高清综合| 久久精品九九亚洲精品| 性做久久久久久久久老女人| 国产国产成人精品久久| 伊人久久无码精品中文字幕| 亚洲国产精品人久久| 亚洲国产欧洲综合997久久| 欧美亚洲日本久久精品| 久久伊人精品青青草原高清| 久久大香香蕉国产| 亚洲AV无码久久精品蜜桃| 亚洲国产婷婷香蕉久久久久久| 91久久精品国产91性色也| 91久久婷婷国产综合精品青草| 久久天天躁狠狠躁夜夜2020一| 青青青青久久精品国产h久久精品五福影院1421| 日韩精品国产自在久久现线拍| 亚洲狠狠婷婷综合久久蜜芽 | 人妻少妇久久中文字幕一区二区| 美女久久久久久| 四虎久久影院| 久久乐国产综合亚洲精品| 最新久久免费视频| 伊人久久无码精品中文字幕| 久久中文字幕人妻熟av女|