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