建議使用MASM for EditPlus進行測試,。
正如"防止程序多重啟動"一文所說,當已經知道自己的實例在運行了,,為了節(jié)省資源,,下個實例需要退出,,可是用戶需要打開文件,,如果不處理就退出,,那用戶不破口大罵才怪,所以,,就需要退出之前進行處理,。
首先要講一個小知識,當Windows程序退出的時候,,Windows會自己清除當前進程所占用的所有資源,除了動態(tài)DLL(靜態(tài)DLL也會自動釋放,,對于運行時動態(tài)加載的DLL,,釋放工作是由用戶完成的,盡管Windows會檢查它是否已經沒有被使用,,但是有時候由于特殊的原因它將不會被釋放),,這就是一個操作系統(tǒng)所謂的資源回收(忘了英文單詞,,好像是這么說吧),那么,,問題就來了,,當程序在退出的時候,盡管可以向其它程序(如自身的實例)發(fā)送消息,,但是問題是如何傳遞資源,,在這里是傳遞一個字符串,而傳遞的資源是又當前進程中的地址,,按Windows的資源回收來說,,當程序退出時傳遞的字符串地址已經不存在了,那如何才能傳遞一個正確的字符串給前一實例呢,?
我們可以使用API中的消息WM_SETTEXT與WM_GETTEXT,這兩個消息發(fā)送后是立即返回的,,這樣當前進程能在發(fā)送完消息后立即退出,而SendMessage函數在發(fā)送消息時會判斷當前消息是否是WM_SETTEXT與WM_GETTEXT,,如果是,,則SendMessage并不單單發(fā)送一個消息,而會創(chuàng)建一個內存映像(標準叫映射)文件,,把需要傳遞的字符保存到這個內存映像中,,然后再把內存映像的地址做為參數傳給目標窗體,而內存映像是可以在程序之間共享的,,這樣就間接的做到了傳遞資源:
... ...
.DATA
szClassName db "WinASM_Class",0
.data?
lpCommandLine DD ?
.CODE
START:
... ...
invoke GetCommandLine
mov lpCommandLine, eax
invoke FindWindow,offset szClassName,NULL ;查找自身類
cmp eax,0
jnz @F
invoke WinMain....
invoke ExitProcess,0
@@:
invoke SendMessage,eax,WM_SETTEXT,0,lpCommandLine
invoke ExitProcess,1 ;記得發(fā)生錯誤后返回非零值,盡管這個不是錯誤,。
......
因此,我們還需要自己來處理WM_SETTEXT消息:
... ...
.elseif uMsg==WM_SETTEXT
invoke MessageBox,0,lParam,0,0
在模板相關地方添加上述代碼后,,運行一下,,你會發(fā)現第二個實例會馬上退出,但是第一個實例會在第二個實例退出的那一瞬彈出一個信息框,,如果第二個實例啟動時你傳遞了參數,,那么你對照一下,信息框中的信息就是第二個實例的尾部參數,。
這樣你就可以在WM_SETTEXT消息中進行處理了,,如打開實例傳過來的文件。