在Win16環(huán)境中,DLL的全局?jǐn)?shù)據(jù)對(duì)每個(gè)載入它的進(jìn)程來(lái)說(shuō)都是相同的,因?yàn)樗械倪M(jìn)程用的都收同一塊地址空間;而在Win32環(huán)境中,情況卻發(fā)生了變化,每個(gè)進(jìn)程都有了它自己的地址空間,DLL函數(shù)中的代碼所創(chuàng)建的任何對(duì)象(包括變量)都?xì)w調(diào)用它的進(jìn)程所有。當(dāng)進(jìn)程在載入DLL時(shí),操作系統(tǒng)自動(dòng)把DLL地址映射到該進(jìn)程的私有空間,也就是進(jìn)程的虛擬地址空間,而且也復(fù)制該DLL的全局?jǐn)?shù)據(jù)的一份拷貝到該進(jìn)程空間。(在物理內(nèi)存中,多進(jìn)程載入DLL時(shí),DLL的代碼段實(shí)際上是只加載了一次,只是將物理地址映射到了各個(gè)調(diào)用它的進(jìn)程的虛擬地址空間中,而全局?jǐn)?shù)據(jù)會(huì)在每個(gè)進(jìn)程都分別加載)。也就是說(shuō)每個(gè)進(jìn)程所擁有的相同的DLL的全局?jǐn)?shù)據(jù),它們的名稱(chēng)相同,但其值卻并不一定是相同的,而且是互不干涉的。
因此,在Win32環(huán)境下要想在多個(gè)進(jìn)程中共享數(shù)據(jù),就必須進(jìn)行必要的設(shè)置。在訪(fǎng)問(wèn)同一個(gè)Dll的各進(jìn)程之間共享存儲(chǔ)器是通過(guò)存儲(chǔ)器映射文件技術(shù)實(shí)現(xiàn)的。也可以把這些需要共享的數(shù)據(jù)分離出來(lái),放置在一個(gè)獨(dú)立的數(shù)據(jù)段里,并把該段的屬性設(shè)置為共享。必須給這些變量賦初值,否則編譯器會(huì)把沒(méi)有賦初始值的變量放在一個(gè)叫未被初始化的數(shù)據(jù)段中。
在DLL的實(shí)現(xiàn)文件中添加下列代碼:
#pragma data_seg("DLLSharedSection") // 聲明共享數(shù)據(jù)段,并命名該數(shù)據(jù)段
int SharedData = 123; // 必須在定義的同時(shí)進(jìn)行初始化!!!!
#pragma data_seg()
在#pragma data_seg("DLLSharedSection")和#pragma data_seg()之間的所有變量將被訪(fǎng)問(wèn)該Dll的所有進(jìn)程看到和共享。僅定義一個(gè)數(shù)據(jù)段還不能達(dá)到共享數(shù)據(jù)的目的,還要告訴編譯器該段的屬性,有三種方法可以實(shí)現(xiàn)該目的(其效果是相同的),一種方法是在.DEF文件中加入如下語(yǔ)句:
SETCTIONS
DLLSharedSection READ WRITE SHARED
另一種方法是在項(xiàng)目設(shè)置的鏈接選項(xiàng)(Project Setting --〉Link)中加入如下語(yǔ)句:
/SECTION:DLLSharedSection,rws
還有一種就是使用指令:
#pragma comment(linker,"/section:.DLLSharedSection,rws")
那么這個(gè)數(shù)據(jù)節(jié)中的數(shù)據(jù)可以在所有DLL的實(shí)例之間共享了。所有對(duì)這些數(shù)據(jù)的操作都針對(duì)同一個(gè)實(shí)例的,而不是在每個(gè)進(jìn)程的地址空間中都有一份。
當(dāng)進(jìn)程隱式或顯式調(diào)用一個(gè)動(dòng)態(tài)庫(kù)里的函數(shù)時(shí),系統(tǒng)都要把這個(gè)動(dòng)態(tài)庫(kù)映射到這個(gè)進(jìn)程的虛擬地址空間里。這使得DLL成為進(jìn)程的一部分,以這個(gè)進(jìn)程的身份執(zhí)行,使用這個(gè)進(jìn)程的堆棧。
下面來(lái)談一下在具體使用共享數(shù)據(jù)段時(shí)需要注意的一些問(wèn)題:
· 所有在共享數(shù)據(jù)段中的變量,只有在數(shù)據(jù)段中經(jīng)過(guò)了初始化之后,才會(huì)是進(jìn)程間共享的。如果沒(méi)有初始化,那么進(jìn)程間訪(fǎng)問(wèn)該變量則是未定義的。
· 所有的共享變量都要放置在共享數(shù)據(jù)段中。如何定義很大的數(shù)組,那么也會(huì)導(dǎo)致很大的DLL。
· 不要在共享數(shù)據(jù)段中存放進(jìn)程相關(guān)的信息。Win32中大多數(shù)的數(shù)據(jù)結(jié)構(gòu)和值(比如HANDLE)只在特定的進(jìn)程上下文中才是有效地。
· 每個(gè)進(jìn)程都有它自己的地址空間。因此不要在共享數(shù)據(jù)段中共享指針,指針指向的地址在不同的地址空間中是不一樣的。
· DLL在每個(gè)進(jìn)程中是被映射在不同的虛擬地址空間中的,因此函數(shù)指針也是不安全的。
當(dāng)然還有其它的方法來(lái)進(jìn)行進(jìn)程間的數(shù)據(jù)共享,比如文件內(nèi)存映射等,這就涉及到通用的進(jìn)程間通信了,這里就不多講了。
轉(zhuǎn)自:
http://m.shnenglu.com/suiaiguo/archive/2009/07/21/90734.html