• <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
            轉自:
            http://topic.csdn.net/t/20020917/11/1030014.html
            使應用程序只能運行一個實例  
                 
                    Windows是多進程操作系統,框架生成的應用程序可以多次運行,形成多個運行實例。  
              但在有些情況下為保證應用程序的安全運行,要求程序只能運行一個實例,比如程    
              序要使用只能被一個進程單獨使用的特殊硬件(例如調制解調器)時,必須限制程    
              序只運行一個實例。  
               
              這里涉及兩個基本的問題,一是在程序的第二個實例啟動時,如何發現該程序已有    
              一個實例在運行,而是如何將第一個實例激活,而第二個實例退出。  
               
              對于第一個問題,可以通過給應用程序設置信號量,實例啟動時首先檢測該信號量,    
              如已存在,則說明程序已運行一個實例。  
               
              第二個問題的難點是獲取第一個實例的主窗對象指針或句柄,然后便可用    
              SetForegroundWindow來激活。雖然FindWindow函數能尋找正運行著的窗口,但該函    
              數要求指明所尋找窗口的標題或窗口類名,不是實現通用方法的途徑。  
              我們可以用Win32   SDK函數SetProp來給應用程序主窗設置一個特有的標記。  
              用GetDesktopWindow   可以獲取Windows系統主控窗口對象指針或句柄,所有應用程  
              序主窗都可看成該窗口的子窗口,即可用GetWindow函數來獲得它們的對象指針或句  
              柄。用Win32   SDK函數GetProp查找每一應用程序主窗是否包含有我們設置的特定標  
              記便可確定它是否我們要尋找的第一個實例主窗。使第二個實例退出很簡單,只要  
              讓其應用程序對象的InitInstance函數返回FALSE即可。此外,當主窗口退出時,應  
              用RemoveProp函數刪除我們為其設置的標記。  
               
              下面的InitInstance、OnCreate和OnDestroy函數代碼將實現上述的操作:  
               
              BOOL   CEllipseWndApp::InitInstance()    
              {    
              //   用應用程序名創建信號量    
              HANDLE   hSem   =   CreateSemaphore(NULL,   1,   1,   m_pszExeName);    
               
              //   信號量已存在?    
              //   信號量存在,則程序已有一個實例運行    
              if   (GetLastError()   ==   ERROR_ALREADY_EXISTS)    
              {    
              //   關閉信號量句柄    
              CloseHandle(hSem);    
              //   尋找先前實例的主窗口    
              HWND   hWndPrevious   =   ::GetWindow(::GetDesktopWindow(),GW_CHILD);    
              while   (::IsWindow(hWndPrevious))    
              {    
              //   檢查窗口是否有預設的標記?    
              //   有,則是我們尋找的主窗    
              if   (::GetProp(hWndPrevious,   m_pszExeName))    
              {    
              //   主窗口已最小化,則恢復其大小    
              if   (::IsIconic(hWndPrevious))    
              ::ShowWindow(hWndPrevious,SW_RESTORE);    
               
              //   將主窗激活    
              ::SetForegroundWindow(hWndPrevious);    
               
              //   將主窗的對話框激活    
              ::SetForegroundWindow(    
              ::GetLastActivePopup(hWndPrevious));    
               
              //   退出本實例    
              return   FALSE;    
              }    
              //   繼續尋找下一個窗口    
              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;    
              //   設置尋找標記    
              ::SetProp(m_hWnd,   AfxGetApp()->m_pszExeName,   (HANDLE)1);    
              return   0;    
              }    
               
              void   CEllipseWndDlg::OnDestroy()      
              {    
              CDialog::OnDestroy();    
              //   刪除尋找標記    
              ::RemoveProp(m_hWnd,   AfxGetApp()->m_pszExeName);      
              }  
                       
               
              對以上代碼的補充:    
                              查看代碼和VC的幫助后,發現問題在于原文在創建信號量和設置尋找標記時使用  
              的是   CWinApp   的成員變量   m_pszExeName,該成員變量其實是應用程序執行文件的名  
              稱去掉擴展名后的部分,而不是應用程序名。    
               
              真正的應用程序名應為成員變量   m_pszAppName,    
              于是將用到m_pszExeName的三處代碼均改為m_pszAppName,重新編譯執行,情況消失。    
               
              最后再提供一個方法和一個信息:    
               
                      另一種使應用程序只能運行一個實例的方法,只需在InitInstance()的最開始添  
              加下列語句即可:    
              HANDLE   m_hMutex=CreateMutex(NULL,TRUE,   m_pszAppName);    
              if(GetLastError()==ERROR_ALREADY_EXISTS)   {   return   FALSE;   }  
               
              但這種方法的不足之處是不能將已經啟動的實例激活。    
               
                            在stingray公司整理開發的MFCFAQ軟件中也提供了一些方法。該軟件實際是一個  
              MFC使用技巧的大匯集,對使用MFC極有幫助,各位朋友不妨去stingray公司的主頁下    
              載。     
               
            posted on 2008-11-02 22:37 Sandy 閱讀(1194) 評論(3)  編輯 收藏 引用 所屬分類: C++

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

            我是這樣理解的,不知道可行么?   回復  更多評論
              
            国产亚洲精久久久久久无码77777 国产亚洲精品久久久久秋霞 | WWW婷婷AV久久久影片| 久久久久久无码Av成人影院| 9191精品国产免费久久| 国产成年无码久久久久毛片| 国产精品亚洲综合专区片高清久久久| 久久久亚洲裙底偷窥综合| 国产成人久久精品麻豆一区| 久久综合久久鬼色| 欧美激情精品久久久久久| 国产激情久久久久久熟女老人| 久久99热国产这有精品| 日产精品久久久久久久| 久久精品中文字幕一区| 亚洲国产精品久久66| 国内精品久久久久久久久电影网 | 亚洲AV无码一区东京热久久| 性欧美大战久久久久久久| 久久久WWW免费人成精品| 精品综合久久久久久88小说| 国产成人精品久久| 久久精品中文无码资源站| 久久综合亚洲色HEZYO社区| 无码精品久久久天天影视| 久久人人爽人人爽人人片AV麻豆 | 精品欧美一区二区三区久久久 | 一本久久a久久精品vr综合| 久久青青草原综合伊人| AAA级久久久精品无码区| 久久久久高潮毛片免费全部播放 | 久久久久人妻一区精品性色av| 亚洲精品国产自在久久| 一本久久知道综合久久| 久久婷婷五月综合国产尤物app| 国产精品欧美亚洲韩国日本久久| 国产精品久久久久久久久| 久久99精品久久久久久噜噜| 99久久精品国产毛片| 青青草原1769久久免费播放| 亚洲国产精品久久久久婷婷软件| 曰曰摸天天摸人人看久久久|