• <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>
            春暖花開
            雪化了,花開了,春天來了
            posts - 149,comments - 125,trackbacks - 0
            轉(zhuǎn)自:
            http://topic.csdn.net/t/20020917/11/1030014.html
            使應(yīng)用程序只能運行一個實例  
                 
                    Windows是多進(jìn)程操作系統(tǒng),框架生成的應(yīng)用程序可以多次運行,形成多個運行實例。  
              但在有些情況下為保證應(yīng)用程序的安全運行,要求程序只能運行一個實例,比如程    
              序要使用只能被一個進(jìn)程單獨使用的特殊硬件(例如調(diào)制解調(diào)器)時,必須限制程    
              序只運行一個實例。  
               
              這里涉及兩個基本的問題,一是在程序的第二個實例啟動時,如何發(fā)現(xiàn)該程序已有    
              一個實例在運行,而是如何將第一個實例激活,而第二個實例退出。  
               
              對于第一個問題,可以通過給應(yīng)用程序設(shè)置信號量,實例啟動時首先檢測該信號量,    
              如已存在,則說明程序已運行一個實例。  
               
              第二個問題的難點是獲取第一個實例的主窗對象指針或句柄,然后便可用    
              SetForegroundWindow來激活。雖然FindWindow函數(shù)能尋找正運行著的窗口,但該函    
              數(shù)要求指明所尋找窗口的標(biāo)題或窗口類名,不是實現(xiàn)通用方法的途徑。  
              我們可以用Win32   SDK函數(shù)SetProp來給應(yīng)用程序主窗設(shè)置一個特有的標(biāo)記。  
              用GetDesktopWindow   可以獲取Windows系統(tǒng)主控窗口對象指針或句柄,所有應(yīng)用程  
              序主窗都可看成該窗口的子窗口,即可用GetWindow函數(shù)來獲得它們的對象指針或句  
              柄。用Win32   SDK函數(shù)GetProp查找每一應(yīng)用程序主窗是否包含有我們設(shè)置的特定標(biāo)  
              記便可確定它是否我們要尋找的第一個實例主窗。使第二個實例退出很簡單,只要  
              讓其應(yīng)用程序?qū)ο蟮腎nitInstance函數(shù)返回FALSE即可。此外,當(dāng)主窗口退出時,應(yīng)  
              用RemoveProp函數(shù)刪除我們?yōu)槠湓O(shè)置的標(biāo)記。  
               
              下面的InitInstance、OnCreate和OnDestroy函數(shù)代碼將實現(xiàn)上述的操作:  
               
              BOOL   CEllipseWndApp::InitInstance()    
              {    
              //   用應(yīng)用程序名創(chuàng)建信號量    
              HANDLE   hSem   =   CreateSemaphore(NULL,   1,   1,   m_pszExeName);    
               
              //   信號量已存在?    
              //   信號量存在,則程序已有一個實例運行    
              if   (GetLastError()   ==   ERROR_ALREADY_EXISTS)    
              {    
              //   關(guān)閉信號量句柄    
              CloseHandle(hSem);    
              //   尋找先前實例的主窗口    
              HWND   hWndPrevious   =   ::GetWindow(::GetDesktopWindow(),GW_CHILD);    
              while   (::IsWindow(hWndPrevious))    
              {    
              //   檢查窗口是否有預(yù)設(shè)的標(biāo)記?    
              //   有,則是我們尋找的主窗    
              if   (::GetProp(hWndPrevious,   m_pszExeName))    
              {    
              //   主窗口已最小化,則恢復(fù)其大小    
              if   (::IsIconic(hWndPrevious))    
              ::ShowWindow(hWndPrevious,SW_RESTORE);    
               
              //   將主窗激活    
              ::SetForegroundWindow(hWndPrevious);    
               
              //   將主窗的對話框激活    
              ::SetForegroundWindow(    
              ::GetLastActivePopup(hWndPrevious));    
               
              //   退出本實例    
              return   FALSE;    
              }    
              //   繼續(xù)尋找下一個窗口    
              hWndPrevious   =   ::GetWindow(hWndPrevious,GW_HWNDNEXT);  
              }    
              //   前一實例已存在,但找不到其主窗    
              //   可能出錯了    
              //   退出本實例    
              return   FALSE;    
              }    
              AfxEnableControlContainer();    
              //   Standard   initialization    
              //   If   you   are   not   using   these   features   and   wish   to   reduce   the   size    
              //   of   your   final   executable,   you   should   remove   from   the   following    
              //   the   specific   initialization   routines   you   do   not   need.    
              #ifdef   _AFXDLL    
              Enable3dControls();   //   Call   this   when   using   MFC   in   a   shared   DLL    
              #else    
              Enable3dControlsStatic();//   Call   this   when   linking   to   MFC   statically    
              #endif    
               
              CEllipseWndDlg   dlg;    
              m_pMainWnd   =   &dlg;    
              int   nResponse   =   dlg.DoModal();    
              if   (nResponse   ==   IDOK)    
              {    
              //   TODO:   Place   code   here   to   handle   when   the   dialog   is    
              //   dismissed   with   OK    
              }    
              else   if   (nResponse   ==   IDCANCEL)    
              {    
              //   TODO:   Place   code   here   to   handle   when   the   dialog   is    
              //   dismissed   with   Cancel    
              }    
              //   Since   the   dialog   has   been   closed,   return   FALSE   so   that   we   exit   the    
              //   application,   rather   than   start   the   application's   message   pump.    
              return   FALSE;    
              }    
               
              int   CEllipseWndDlg::OnCreate(LPCREATESTRUCT   lpCreateStruct)      
              {    
              if   (CDialog::OnCreate(lpCreateStruct)   ==   -1)    
              return   -1;    
              //   設(shè)置尋找標(biāo)記    
              ::SetProp(m_hWnd,   AfxGetApp()->m_pszExeName,   (HANDLE)1);    
              return   0;    
              }    
               
              void   CEllipseWndDlg::OnDestroy()      
              {    
              CDialog::OnDestroy();    
              //   刪除尋找標(biāo)記    
              ::RemoveProp(m_hWnd,   AfxGetApp()->m_pszExeName);      
              }  
                       
               
              對以上代碼的補(bǔ)充:    
                              查看代碼和VC的幫助后,發(fā)現(xiàn)問題在于原文在創(chuàng)建信號量和設(shè)置尋找標(biāo)記時使用  
              的是   CWinApp   的成員變量   m_pszExeName,該成員變量其實是應(yīng)用程序執(zhí)行文件的名  
              稱去掉擴(kuò)展名后的部分,而不是應(yīng)用程序名。    
               
              真正的應(yīng)用程序名應(yīng)為成員變量   m_pszAppName,    
              于是將用到m_pszExeName的三處代碼均改為m_pszAppName,重新編譯執(zhí)行,情況消失。    
               
              最后再提供一個方法和一個信息:    
               
                      另一種使應(yīng)用程序只能運行一個實例的方法,只需在InitInstance()的最開始添  
              加下列語句即可:    
              HANDLE   m_hMutex=CreateMutex(NULL,TRUE,   m_pszAppName);    
              if(GetLastError()==ERROR_ALREADY_EXISTS)   {   return   FALSE;   }  
               
              但這種方法的不足之處是不能將已經(jīng)啟動的實例激活。    
               
                            在stingray公司整理開發(fā)的MFCFAQ軟件中也提供了一些方法。該軟件實際是一個  
              MFC使用技巧的大匯集,對使用MFC極有幫助,各位朋友不妨去stingray公司的主頁下    
              載。     
               
            posted on 2008-11-02 22:37 Sandy 閱讀(1195) 評論(3)  編輯 收藏 引用 所屬分類: C++

            FeedBack:
            # re: 轉(zhuǎn):使應(yīng)用程序只能運行一個實例 [未登錄]
            2008-11-03 12:49 | SpringSnow
            這個方法是通過設(shè)置標(biāo)志量的,是一種不錯的思考方法,值得借鑒。在自己編寫的應(yīng)用程序中,是簡單的通過FindWindow來實現(xiàn)的。還沒有做過什么比較。這個問題值得去思考一下。
            如何使應(yīng)用程序只運行一個實例,有多少種方法,各有什么利弊呢!  回復(fù)  更多評論
              
            # re: 轉(zhuǎn):使應(yīng)用程序只能運行一個實例
            2009-06-11 23:06 | Brandon
            如果是非窗口程序呢?  回復(fù)  更多評論
              
            # re: 轉(zhuǎn):使應(yīng)用程序只能運行一個實例
            2009-06-12 13:59 | Sandy
            @Brandon
            應(yīng)該是一樣的吧,可以借助信號量.
            如果是非窗口的,就不用將窗口置前了,直接就退出了.

            我是這樣理解的,不知道可行么?   回復(fù)  更多評論
              
            国产亚洲精久久久久久无码AV| 久久青青草视频| 久久久久国产精品熟女影院| 综合久久久久久中文字幕亚洲国产国产综合一区首 | 亚州日韩精品专区久久久| 伊人久久大香线蕉精品不卡| 99久久国产宗和精品1上映| 亚洲精品美女久久久久99| 久久国产乱子精品免费女| 久久乐国产综合亚洲精品| 国内精品久久久久久99蜜桃| 国产免费久久精品丫丫| 伊人久久大香线蕉av不卡 | 久久影院午夜理论片无码| 亚洲精品无码久久久久sm| 日韩中文久久| 99久久亚洲综合精品网站| 久久人人添人人爽添人人片牛牛| 97久久超碰国产精品2021| 伊人久久一区二区三区无码| 99久久精品九九亚洲精品| 久久精品国产亚洲AV蜜臀色欲| 99久久国产亚洲高清观看2024| 国产69精品久久久久久人妻精品| 2020最新久久久视精品爱| 久久精品黄AA片一区二区三区| 久久无码一区二区三区少妇| 色综合久久88色综合天天| 久久国产色AV免费看| 亚洲AV无码久久寂寞少妇| 色天使久久综合网天天 | 国产高潮久久免费观看| 久久大香香蕉国产| 久久精品水蜜桃av综合天堂| 久久99久久99精品免视看动漫| 久久久综合香蕉尹人综合网| 精品人妻伦九区久久AAA片69| 精品综合久久久久久88小说| 国产精品99久久久久久宅男 | 2021最新久久久视精品爱| 亚洲?V乱码久久精品蜜桃|