【限制程式只開啟一個例項】

fictiony發表於2004-04-12
【限制程式只開啟一個例項】
【作者:Fictiony (fictiony@china.com)】
【轉載請註明出處】
  當我們在做一些管理平臺類的程式(比如Windows的工作管理員)時,往往需要限制程式只能開啟一個例項。解決這個問題的大致思路很簡單,無非是在程式開啟的時候判斷一下是否有與自己相同的程式開著,如果有,則關閉自身,否則正常執行。
  但是,問題就出在如何判別是否有一個與自己相同的程式開著上面。我在網上搜尋了一下相關的文章,發現對於這個問題的解決不外乎以下幾種方式:
  1、在程式初始化時使用::CreateMutex建立一個互斥物件,通過檢測互斥物件是否已存在來確定該程式是否已經執行。
  該方式的確可以很容易的實現判別程式例項是否已存在,只需要在InitInstance方法開頭新增以下語句:
  UNIQUE_ID為具有唯一性的字串,一般可以用VC++為主程式標頭檔案自動生成的包含標識巨集(就是.h檔案頂上的那一長串巨集定義),當然,也可以用工具自己手動生成,隨君所好了^^。要注意的是別忘了在ExitInstance方法中用 CloseHandle(m_hUnique) 將該互斥物件關閉。但這種方式存在一個很大的問題,就是很難獲取已開啟程式例項的主視窗控制程式碼。而我們絕大多數時候,都需要將那個程式例項的主視窗啟用。為了獲取主視窗控制程式碼,就需要再用到後面提到的其他方法。
  2、遍歷所有已經開啟的程式主視窗,比較視窗標題,如果找到滿足條件的標題,則表示程式已經執行,並啟用該視窗。
  這種方式雖然可以找到程式的主視窗,但問題明顯:A.如果視窗標題經常變化怎麼辦(比如標題中會帶有開啟文件的檔名)?B.如果其他程式的主視窗標題恰好與該程式的相同怎麼辦?
  第一個問題可以通過寫登錄檔或者寫INI檔案的方式來解決。即當主視窗標題改變時,將新標題寫入登錄檔或者INI檔案。不過這種解決方式也忒麻煩了吧-_-||   第二個問題就麻煩了,至少我還沒有找到好的解決方案。如果非要說一個,那我提議你“想盡辦法”“不擇手段”的將視窗標題設的和別的程式絕對不同。不過估計搞定了這步,你半條命也快沒了。
  3、用::SetProp給主視窗新增一個具有唯一性的屬性值,以便在程式初始化的時候可以通過遍歷所有視窗的該屬性來判斷。
  新增屬性值的程式碼一般可以放在InitInstance方法的最後,如下:
  UNIQUE_ID是一個具有唯一性的整數值(為什麼不能用字串?因為字串的比較需要將字串讀取出來,而這兒只能記錄字串地址,在別的程式裡這個地址無意義,所以無法讀出這個字串)。這種方式僅有的問題就出在如何確定該整數值是具有唯一性的。我們後面提出的解決方法,就是在這種方法的基礎上發展出來的。
  在總結了上述幾種方式的利弊之後,我發現,只需要為程式建立一個具有唯一性的整數值,一方面可以通過這個值是否存在來判斷程式是否已經執行(::CreateMutex其實也是類似的概念),另一方面可以通過將這個值賦給主視窗,以便能夠找到已開啟的程式例項的主視窗控制程式碼。於是,ATOM量便派上用場了(ATOM變數型別等同於WORD,因而是一個整數值)。
  ATOM量本質上就是雜湊表的鍵識別符號,其對應鍵值為一個字串。每個程式都有自己的ATOM量表,同時Windows也有一個全域性的ATOM表。我們要用的方法就是,為程式建立一個全域性的ATOM量,通過這個量是否存在來判斷程式是否已經執行,並通過將這個量作為屬性值新增到主視窗來標識這個主視窗。具體過程如下:
  1、給主程式App類新增一個ATOM型別的成員變數:m_aAppId,作為程式ID。
  2、在InitInstance方法開頭新增以下程式碼(UNIQUE_ID是具有唯一性的字串巨集):
  3、在InitInstance方法最後為主視窗新增標識屬性:
  4、在ExitInstance方法中新增下面程式碼以刪除程式ID:
  心得:該方法所用到的ATOM量是一個應用廣泛的技術,如::CreateMutex、::SetProp等API函式都間接用到了ATOM量。利用它,我們可以做很多需要用到唯一性驗證的事情。

相關文章