青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

yehao's Blog

防止C++程序重復運行的幾種方法

轉自http://hi.baidu.com/dreamyguy/blog/item/aacc6f44086afe45500ffef9.html

有時候,為了某些要求,我們希望程序實例只運行一次。而在VB6中,我們可以很輕易的根據App.hPreInstance來判斷程序是否已經運行。但是在C++中,這一切就變得不是那么容易。

雖然WinMain函數有hPreInstance參數來指示,但是那是在Win16位的前提下,到了32Bit時代,那個參數已經完全成為擺設。

而本文正好探討了如何防止C++程序重復運行的方法。

PS:因為本人使用MFC,所以為了方便,所有代碼均為以MFC為基礎。大家可以根據自己需要更改

  1.查找窗體

對于存在GUI窗體(CUI暫不討論)的程序來說,最容易想到的就是利用FindWindow,以標題作為參數進行查找主窗體,然后使其關閉即可。

通常,我們能寫出如下代碼:

// Find Window by Caption
// Add this code in InitInstance function of class 
// you have derived from the CWinApp class

HWND hWnd 
= ::FindWindow(NULL, "MFCDialog");

if (hWnd)
{
AfxMessageBox(
"Has been running");
return FALSE;
}

 

以上的代碼可以簡單的起到防止重復啟動的效果,但是局限性很大。

首先,由于在FindWindow中要指定窗體的標題,如果窗體的標題在程序運行中是不斷變化的,那么就給搜索帶來了一定難度。

而且,如果其他程序也恰好是用相同的標題的話……- -#。當然,你可以通過在FindWindow中指定類名來減少錯誤。但是如果你看過我前面寫的文章的話,你就會發現,MFC注冊窗口類并不是那么隨意,而是經過N次陰謀籌劃之后……

看來這方法的局限性的確很大- -#

  2.額外窗體存儲

此方法來源于對上面一種方法的補充,因為通過搜索MFC的窗體類比較困難,而且準確度不一定高。所以,我想到了使用額外窗體存儲(Extra Widnow Memory)的方法

PS:關于什么是額外窗體存儲,請自行google或MSDN或查看我曾經寫的The Analyses Of Windows Runnning Principle

如果你使用SDK進行開發,可以在創建窗體時填充這一屬性,然后用GetWindowLong獲取。

而由于我使用MFC,所以我更關注如何在MFC中使用這一屬性。

一般來說,我們可以使用SetWindowLong對額外窗體存儲進行填充,然后用GetWindowLong獲取,最后配合FindWindow來檢驗程序是否重復運行。

// Add this code in InitDialog function
// and you can specify any number you want
BOOL bRet = ::SetWindowLong(GetSafeHwnd(), GWL_USERDATA, 256);

// Add this code in InitInstance function
// Find Window by using extra memory

HWND hWnd 
= FindWindow(NULL, "MFCDialog");

if (hWnd)
{
BOOL bRet 
= ::GetWindowLong(hWnd, GWL_USERDATA);

if (256 == bRet)  // compare
    {
AfxMessageBox(
"Has been running");
return FALSE;
}
}

 

 3.全局原子

你可以使用GlobalAddAtom將某個特定的字符串添加到全局原子列表(Global Atom Table),然后在程序運行時檢查該字符串即可。

但是這個方法有一個致命的弱點,程序退出時,Windows不會自動為你刪除添加到列表中的Atom,而是需要你自己使用GlobalDeleteAtom進行刪除。

這就意味著,如果你的程序意外的退出了,沒有刪除添加的Atom,那么,你的程序將無法運行。

所以,這并不是一個好方法。


  4.枚舉進程

這或許是一個畢竟正常,或者說相對穩定的方法。

我們可以使用CreateToolhelp32Snapshot或者EnumProcess來枚舉當前的進程,然后檢查是否已經運行。如果擔心存在同名的進程,還可以檢查路徑。

//   關閉一個進程  
static   bool   CloseApplication(CString   sAppToClose,   CString   sAppToUpdate,    
CString   sFilename,   bool   bAskUser   /*   =   true   */)  
{  
//   Declare   variables  
HANDLE   hProcessSnap   =   NULL;  
PROCESSENTRY32   pe32   =   {NULL};  
CString   sCompare,   sTemp;  

       //sProcessFileName是你的進程名  
CString   sProcesFilename   =   m_pPath->ExtractFilename(sFilename);  
CLanguage   *   pLanguage   =   CLanguage::Instance();  

//調整統一大小寫格式,方便比較、  
#if   (_MFC_VER   <   0x0700)  
sProcesFilename.MakeLower();  
#else  
CString   sLowerTemp;  
sLowerTemp   =   sProcesFilename;  
sProcesFilename   =   sLowerTemp.MakeLower();  
#endif  

hProcessSnap   =   CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,   0);  

pe32.dwSize   =   sizeof(PROCESSENTRY32);  
//   系統進程快照  
if (Process32First(hProcessSnap,   &pe32))  
{  
do  
{  
sCompare.Format("%s",   pe32.szExeFile);  

#if   (_MFC_VER   <   0x0700)  
sCompare.MakeLower();  
#else  
CString   sLowerTemp;  
sLowerTemp   =   sCompare;  
sCompare   =   sLowerTemp.MakeLower();  
#endif  


//   Check   if   we   found   the   right   process  
if(sCompare   ==   sProcesFilename)  
{  
//   If   we   are   not   in   hidden   mode  
if(bAskUser)  
{  
//Set   up   confirmation   text  
sTemp.Format(pLanguage->GetString(IDS_CONFIRMATION_CLOSEAPPLICATION_EXPLANATION),  
sAppToClose,   sAppToUpdate,   sAppToClose);  

//Ask   confirmation  
CConfirmationDlg   dlgConfirm(pLanguage->GetString(IDS_CONFIRMATION_CLOSEAPPLICATION_TITLE),  
pLanguage->GetString(IDS_CONFIRMATION_TITLE),   sTemp,  
pLanguage->GetString(IDS_GENERAL_YES),   pLanguage->GetString(IDS_GENERAL_NO));  
if(dlgConfirm.DoModal()   ==   IDNO)  
{  
return   false;  
}  
}  

//   Get   handle   to   process  
HANDLE   hProcess   =   OpenProcess(PROCESS_ALL_ACCESS,   FALSE,   pe32.th32ProcessID);  

//   Exit   process  
DWORD   exCode;  
GetExitCodeProcess(hProcess,   &exCode);  
TerminateProcess(hProcess,   exCode);  

//   Wait   until   process   is   finished  
if   (WaitForSingleObject(hProcess,   30000)   ==   WAIT_TIMEOUT)  
return   false;  
}  
}   while   (Process32Next(hProcessSnap,   &pe32));  
}  
else  
{  
return   false;  
}  

return   true;  
}

PS:在Vista下使用EnumProcess時,要注意權限問題,OpenProcess增加了一個新的權限常數,僅限Vista。如果不增加這個參數,很多進程是無法被枚舉出來的(不過MS不印象我們自己的進程- -#)。


  5.互斥對象

使用互斥對象來防止程序重復運行是一個很常用的做法,而且M$也推薦使用這種方法。和上面的幾種方法相比,需要寫的代碼少,而且效率比較高。所謂方便易用~

一般我們會使用CreateMutex來創建互斥體,當第二次創建相同的互斥體時,這個API會返回前一個互斥體的Handle,而GetLastError則會返回ERROR_ALREADY_EXITS

// Mutex Object 
// Add this code in InistInstance function
  
HANDLE hMutex 
= NULL;
TCHAR 
*lpszName = "TestMutex";

hMutex 
= ::CreateMutex(NULL, FALSE, lpszName);
DWORD dwRet 
= ::GetLastError();

if (hMutex)
{
if (ERROR_ALREADY_EXISTS == dwRet)
{
AfxMessageBox(
"Has been running");
CloseHandle(hMutex);  
// should be closed
      return FALSE;
}
}
else
{
AfxMessageBox(
"Create Mutex Error");
}

// Add this code in Destruction function
  
::CloseHandle(hMutex);

使用互斥體時要注意幾個問題:
在CreateMutex之后馬上GetLastError,GetLastError是一個很復雜的API,任何牽涉到GetLastError的操作在執行之后,都會覆蓋先前的值。

把正常的CloseHandle寫到窗體的析構函數或者程序對象的析構函數里。不要在CreateMutex之后立刻CloseHandle,否則互斥對象會被清空。

這也是我當初所犯的錯誤,(不知道網上那么多錯誤的代碼是不是經過Debug的囧),當互斥對象的最后一個Handle被Close之后,互斥對象將被刪除。如果程序在退出時沒有清空互斥對象,Windows將會執行這一操作。當然,把次操作交給OS不是一個好習慣。

詳情請看MSDN的引用:
引用:
Use the CloseHandle function to close the handle. The system closes the handle automatically when the process terminates. The mutex object is destroyed when its last handle has been closed.
轉自:http://bbs.cfan.com.cn/viewthread.php?tid=793295
http://topic.csdn.net/t/20060424/09/4708145.html
http://student.csdn.net/space.php?uid=110004&do=thread&id=3278

posted on 2011-05-03 18:28 厚積薄發 閱讀(744) 評論(0)  編輯 收藏 引用 所屬分類: Windows編程

導航

<2013年6月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
30123456

統計

常用鏈接

留言簿

隨筆分類

文章分類

文章檔案

搜索

最新評論

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美视频不卡中文| 国产在线播放一区二区三区| 夜夜狂射影院欧美极品| 欧美成年人视频网站| 久久亚洲影院| 另类尿喷潮videofree| 免费亚洲一区| 亚洲级视频在线观看免费1级| 欧美另类综合| 欧美电影美腿模特1979在线看 | 亚洲无玛一区| 翔田千里一区二区| 欧美一区二区三区在线看| 欧美一二三区在线观看| 久久国产精品99国产精| 欧美1区2区| 伊人久久亚洲热| 亚洲午夜羞羞片| 久久精品国产亚洲aⅴ| 久久中文字幕一区| 欧美日韩一区在线视频| 国产亚洲精品久久久| 亚洲国产精品久久久| 亚洲线精品一区二区三区八戒| 亚洲欧美日产图| 可以免费看不卡的av网站| 亚洲电影免费观看高清完整版在线| 亚洲成人资源| 欧美一区二区三区播放老司机 | 亚洲国产精品悠悠久久琪琪| 99精品热视频只有精品10| 亚洲欧美日韩天堂一区二区| 美女精品网站| 国产欧美视频一区二区| 亚洲伦理在线免费看| 欧美一区二区三区日韩视频| 蜜臀久久久99精品久久久久久 | av不卡在线| 久久久亚洲精品一区二区三区| 欧美激情1区2区3区| 在线一区二区三区四区| 美女任你摸久久| 国产精品视频男人的天堂| 亚洲精品一区二区三| 久久免费精品视频| 亚洲网站在线播放| 欧美日韩精品免费 | 欧美aaa级| 欧美日韩综合精品| 亚洲精品乱码久久久久久黑人| 亚洲男人av电影| 亚洲精品一区在线| 久久综合给合| 国产一区在线看| 欧美中在线观看| 亚洲一级电影| 国产精品久久久久永久免费观看| 亚洲国产精品传媒在线观看| 久久精品国产精品| 亚洲在线一区二区| 99精品热视频| 国产精品欧美日韩一区| 亚洲欧美在线视频观看| 亚洲茄子视频| 欧美精品一区二| 9色精品在线| 亚洲精品中文字幕在线观看| 欧美国产精品v| 日韩午夜激情| 亚洲精品影视| 欧美日韩性视频在线| 亚洲最新在线| 一区二区日本视频| 国产精品日韩精品| 久久爱91午夜羞羞| 欧美一区二区私人影院日本 | 亚洲人在线视频| 欧美va亚洲va香蕉在线| 久热成人在线视频| 91久久精品网| 日韩视频―中文字幕| 欧美亚洲不卡| 久久精品动漫| 在线观看欧美黄色| 国产一区二区三区高清在线观看 | 99精品国产在热久久| 欧美午夜电影一区| 久久国产欧美日韩精品| 久久婷婷麻豆| 在线中文字幕一区| 欧美一区二区免费视频| 亚洲精品一二区| 亚洲一区二区三区在线| 狠狠综合久久av一区二区小说| 亚洲电影观看| 国产日韩精品在线| 亚洲国产精品第一区二区| 欧美日韩国产一区| 久久嫩草精品久久久久| 欧美另类极品videosbest最新版本| 香蕉久久夜色精品国产使用方法| 久久精选视频| 亚洲一区中文| 欧美18av| 噜噜噜久久亚洲精品国产品小说| 欧美久久在线| 免费观看亚洲视频大全| 欧美色大人视频| 欧美激情一二三区| 国产精品综合| 99re66热这里只有精品3直播 | 亚洲精品在线观| 亚洲欧美在线高清| 精品白丝av| 亚洲一区二区四区| 国内精品国产成人| 亚洲国产视频一区| 久久精品国产亚洲精品| 亚洲三级免费| 性欧美长视频| 欧美在线观看视频一区二区三区 | 欧美一区二区三区免费视| 最新精品在线| 亚洲一区二区精品在线观看| 极品日韩久久| 久久美女性网| 国产乱码精品1区2区3区| 欧美www在线| 国产亚洲第一区| 日韩视频久久| 国内一区二区在线视频观看| 噜噜噜躁狠狠躁狠狠精品视频 | 最新成人av在线| 韩日欧美一区二区| 久久精品夜色噜噜亚洲aⅴ| 国产精品久久久久久久一区探花 | 性欧美xxxx大乳国产app| 亚洲精品美女在线| 亚洲欧美伊人| 亚洲精品久久久久| 欧美xxx在线观看| 久久九九国产| 欧美色精品在线视频| 亚洲国产精品精华液2区45| 国产一级揄自揄精品视频| 午夜精品成人在线视频| 亚洲视频在线观看三级| 国产麻豆精品theporn| 好吊日精品视频| 亚洲色图综合久久| 国产一区二区| 久久综合久色欧美综合狠狠 | 尤物视频一区二区| 欧美一区二区三区在线看| 亚洲综合导航| 欧美视频久久| 日韩视频免费| 99这里只有精品| 欧美日韩精品系列| 亚洲精品视频在线播放| 亚洲美洲欧洲综合国产一区| 免费av成人在线| 性色av一区二区三区| 韩日精品在线| 媚黑女一区二区| 亚洲大胆人体在线| 亚洲美女黄色| 欧美夫妇交换俱乐部在线观看| 欧美二区在线播放| 亚洲国产专区校园欧美| 麻豆av一区二区三区| 国产亚洲一区二区在线观看| 欧美在线视频一区二区| 亚洲综合久久久久| 欧美二区在线| 亚洲一区二区三区777| 午夜精品久久久久久久99水蜜桃| 欧美日韩专区在线| 午夜在线观看欧美| 欧美一级二区| 亚洲毛片播放| 国产精品视频一区二区高潮| 午夜精品剧场| 欧美激情导航| 亚洲影视在线| 黄色国产精品| 黄色成人av网站| 久久影院午夜片一区| 欧美视频亚洲视频| 欧美成人中文字幕| 一区二区三区四区精品| 久久成人18免费观看| 在线精品视频在线观看高清| 国产精品成人观看视频免费| 午夜欧美不卡精品aaaaa| 蜜臀91精品一区二区三区| 一本高清dvd不卡在线观看| 国产精品毛片在线看| 欧美精品午夜| 欧美在线精品免播放器视频|