最近遇到一個不太懂的事,但我覺得應該有不少人寫過并且有一個相對普遍的方法去實作,然而我百度沒有找到,所以在這里請教一下大佬們指導一下小弟,謝謝了
是這樣的
檔案型別已經關聯好了,如果沒啟動程式,雙擊檔案也能啟動程式。
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
std::string targetfile = lpCmdLine; //注冊檔案型別后,雙擊對應檔案型別,這個值將接收目標檔案
//查找主表單是否已經運行
HWND hwnd = ::FindWindow(NULL, _T("MainWnd"));
if (hwnd != NULL) {
//如果用戶通過雙擊打開關聯的檔案,則嘗試打開檔案
if (targetfile.size()) {
//const char * tf = "你就是個god"; //成功
//const char * tf = targetfile.data(); //失敗
//MessageBox(hwnd, tf, NULL, 0);//全部成功,正確顯示雙擊的檔案路徑
::SendMessage(hwnd, WM_OPENTARGETFILE, 0, (LPARAM)tf);
}
return 0;
}
else {
//主表單沒運行,就運行主表單
CMainWnd* pFrame = new CMainWnd;
if (pFrame == NULL) return 0;
pFrame->Create(NULL, _T("MainWnd"), UI_WNDSTYLE_FRAME, 0L, 0, 0, 990, 690);
pFrame->CenterWindow();
pFrame->ShowModal();
delete pFrame;
return 0;
}
}
這傳值,就是把我雙擊那個檔案的路徑傳到這個MainWnd里,這一步就難到我,看上面代碼,我具體賦值是成功傳值的,但用targetfile就不行了,不管是c_str()還是data()全部不能傳,但彈出視窗的時候卻都是正確的。問了一個也是寫代碼的,他比我了解一些,但也沒寫過這種,他說這里應該是多行程的,打開多少個檔案就多少個行程,傳不到值是因為跨行程了,要行程間通信,我想知道,大家在這里是怎么處理的,就是雙擊檔案傳值這里,是多行程通信嗎?還是怎么傳的,虛心求指導!
uj5u.com熱心網友回復:
先要//允許用戶通過Windows的檔案管理器打開資料檔案
EnableShellOpen():
然后自己搜索RegisterShellFileTypes
uj5u.com熱心網友回復:
跨行程,可以考慮使用剪切板uj5u.com熱心網友回復:
WM_COPYDATA 也可以參考下Using Data Copy
uj5u.com熱心網友回復:
你的意思是說,的確是用多行程的,同時打開10檔案,就有10個行程?你做過打開檔案嗎?uj5u.com熱心網友回復:
感謝你的回復,我的程式是win32, 添加了頭檔案afxwin.h,依然不能使用這兩個函式
還有的就是,這幾個函式的放置位置問題,我的入口函式是APIENTRY WinMain,能不能在我的代碼上修改一下,具體一點,如果了解DragAcceptFiles(),看能不能一起改改,多加分
uj5u.com熱心網友回復:
沒太理解你的意思, 多個行程彈出檔案打開對話框?
uj5u.com熱心網友回復:
你的意思是說,的確是用多行程的,同時打開10檔案,就有10個行程?你做過打開檔案嗎?
跨行程,可以考慮使用剪切板
沒太理解你的意思, 多個行程彈出檔案打開對話框?
其實這種情況現實中非常多,隨便撿個說吧。[word也是,photoshop也是,太多了,但大家好像都沒討論過這個,可能是太簡單了]
像.url檔案,是個網址的快捷方式,雙擊就是啟動瀏覽器,然后創建一個標簽頁進行打開,對吧?
好,現在的情況是瀏覽器已經啟動了,并且已經打開了一個網址,沒錯吧?
現在用戶又雙擊了另一個.url檔案,瀏覽器又會通過新建標簽頁打開一個網址,兩個標簽頁, 兩個.url, 對吧?
好了,問題就是,
1,雙擊.url的時候,瀏覽器是如何把接收的這個.url的路徑從程式入口傳到處理函式的呢?SendMessage么?
2,這兩個標簽頁之間,是行程與行程的關系嗎?如果是這樣的話,那傳送.url的路徑就應該是行程間通信,而不是SendMessage了?我這樣理解,正確嗎?
3,如果沒那么復雜,win32有自帶的方法,求具體代碼或鏈接
uj5u.com熱心網友回復:
這種特性, MFC向導生成一個多檔案MDI程式,就支持的。主要的機理就是 Dynamic Data Exchange, WM_DDE_INITIATE WM_DDE_EXECUTE WM_DDE_TERMINATE
uj5u.com熱心網友回復:
這種特性, MFC向導生成一個多檔案MDI程式,就支持的。
主要的機理就是 Dynamic Data Exchange, WM_DDE_INITIATE WM_DDE_EXECUTE WM_DDE_TERMINATE
uj5u.com熱心網友回復:
Using Dynamic Data Exchangeuj5u.com熱心網友回復:
單行程MFC嘛,去開源社區搜索MFC single instance就可以找到好多示例代碼了。MFC雖然自己有EnableShellOpen但是這個是基于已經過時的DDE。新的代碼建議用IDropTarget注冊檔案關聯——用戶選擇了多個檔案然后選擇打開的時候,你的程式只會被啟動一次,選擇的檔案名陣列會在一個shell data object里面傳給你,參考https://devblogs.microsoft.com/oldnewthing/20100503-00/?p=14183當然這個只是針對資源管理器打開檔案,用命令列還是能啟動多個你的程式的實體的。單行程控制的話個人喜歡用共享記憶體通訊,用Mutex做同步。找視窗太不靠譜了,你選中兩個檔案同時打開,會出現找不到視窗的情況。
uj5u.com熱心網友回復:
共享進城間傳址是不行的,需要共享記憶體傳數值過去,或者遠程程序呼叫,傳過去。手機無力,沒法代碼……uj5u.com熱心網友回復:
以前寫的不知道現在行不行
////
BOOL CMainFrame::GetExeCommand(char *fullname,char* command)
{// ""%1" 1*" for .bat file
// "xxx.exe %1"
// ""\a b\c\yyy.exe" /p %1"
char ext[40];
char pathname[MAX_PATH];
char exename[MAX_PATH];
char *at=0;
strcpy(pathname,fullname);
// must ".xxx"
if((at=strrchr(fullname,'.'))==NULL) return FALSE;
strcpy(ext,at);
char open_key[MAX_PATH];
DWORD openkeylen=MAX_PATH;
char shellname[MAX_PATH]="shell";
// under shell may be different "open","cplopen","Open with"...
BYTE extfile[MAX_PATH];
DWORD cbextfile=MAX_PATH;
DWORD dwType=REG_SZ;
HKEY hkey;
HKEY hsubkey;
// from ".xxx" get "xxxfile"
// C:\Users\Shen Hui\AppData\Roaming\Heinote\hnote.exe %1
// HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\. txt
DWORD dwErrorCode=RegOpenKeyEx(HKEY_CLASSES_ROOT,ext,0,KEY_READ,&hkey);
if (dwErrorCode != ERROR_SUCCESS)
{ // error not found
command[0]=0;
return FALSE;
}
else
{ // ".xxx" OK
// get "xxxfile"
dwErrorCode=RegQueryValueEx(hkey,NULL,0,&dwType,extfile,&cbextfile);
RegCloseKey(hkey);
char tmp[MAX_PATH];
sprintf(tmp,"%s\\%s",extfile,shellname);// "cppfile\\shell"
if (dwErrorCode == ERROR_SUCCESS)
{// "cppfile\\shell"
dwErrorCode=RegOpenKeyEx(HKEY_CLASSES_ROOT,tmp,0,KEY_READ,&hsubkey);
if (dwErrorCode == ERROR_SUCCESS)
{// "cppfile\\shell"
LONG enumErr=ERROR_SUCCESS;
DWORD j=0;
FILETIME ftLastWriteTime;
while (enumErr==ERROR_SUCCESS || enumErr==ERROR_MORE_DATA)
{
cbextfile=260;// re-init
openkeylen=MAX_PATH;// re-init
enumErr=RegEnumKeyEx(hsubkey,j,open_key,&openkeylen,
NULL,NULL,NULL,&ftLastWriteTime);
strcpy(tmp,open_key);
_strupr(tmp);
if(strstr(tmp,"OPEN")) // any "open"
{
sprintf(tmp,"%s\\%s\\%s\\command",extfile,shellname,open_key);// "cppfile\\shell"
break;
}
j++;
}
RegCloseKey(hsubkey);
if(enumErr!=ERROR_SUCCESS) return FALSE;
} // end with "shell" key
else
{ // no "shell" key
return FALSE;
}
// "shell\\...open...\\command"
dwErrorCode=RegOpenKeyEx(HKEY_CLASSES_ROOT,tmp,0,KEY_READ,&hsubkey);
cbextfile=260;// re-init
//extfile="%SystemRoot%\\system32\\NOTEPAD32.EXE %1"
dwErrorCode=RegQueryValueEx(hsubkey,NULL,0,&dwType,extfile,&cbextfile);
// if fullname contains space in its pathname ????
// if(strchr(fullname,' ')) sprintf(exename,"\"%s\"",fullname);
// else
strcpy(exename,fullname);
// replace "%SystemRoot%" to "c:\\winnt.400" //NT
if(strstr((char *)extfile,"%SystemRoot%"))
{
char windir[MAX_PATH];
GetWindowsDirectory(windir,MAX_PATH);
at=strchr((char *)extfile,'\\');// after "%SystemRoot%"
strcat(windir,at);
strcpy((char *)extfile,windir);
}
// parameterse
strcpy(tmp,(char*)extfile);
if(at=strstr(tmp,"%1"))
{
*(at+1)='s';// "%s"
sprintf(command,tmp,exename);
}
else
{
sprintf(command,"%s %s",tmp,exename);
}
RegCloseKey(hsubkey);
return TRUE;
}
else
{
command[0]=0;
return FALSE;
}
}
}
uj5u.com熱心網友回復:
呼叫雙擊-》void CMainFrame::OnFileOpen()
{
//////
// other file
if(GetExeCommand(fullname,command))
{ // OK
if(!GoCommand(command))
{
CString prompt;
AfxFormatString1(prompt,IDS_BAD_COMMAND,fullname);
AfxMessageBox(prompt);
}
return;
}
/////////////
轉載請註明出處,本文鏈接:https://www.uj5u.com/gongcheng/13565.html
標籤:進程/線程/DLL
上一篇:UDP和TCP協議
