• ベストアンサー

ShellExecuteEx→WaitForSingleObjectで返ってこない

VC2008にて、エクスプローラもどきを作成しています。 ファイルアイコンをダブルクリックした時に、 関連付けされたアプリケーションで起動し、 そのファイルが閉じられたら、以降の処理を行うという 流れのプログラムを作成しています。 ファイル起動を、ShellExecuteで起動させ、 その後、WaitForSingleObjectを呼んで、 ファイルが閉じられるまで待つようにしています。 しかし、ファイルを閉じてもWaitFor~の次の処理に 来ず、困っています。 以下、ソースの抜粋になります。 ---------------ここから---------------- CString dir; dir = "C:\\test.doc"; SHELLEXECUTEINFO si; ZeroMemory( &si, sizeof(si) ); si.cbSize = sizeof(si); si.fMask = SEE_MASK_NOCLOSEPROCESS; si.lpVerb = _T("open"); si.lpFile = dir; si.nShow = SW_SHOWNORMAL; BOOL rtn; rtn = ShellExecuteEx( &si ); if ( rtn == 0 || (const int)si.hInstApp <= 32 ){ MessageBox( "open error" ); return; } WaitForSingleObject( si.hProcess, INFINITE ); CloseHandle( si.hProcess ); MessageBox( "次の処理start" ); ---------------ここまで---------------- ここでは、wordファイルをテストしているのですが、 ファイルを閉じても、完全にwordが終了せず、 2~3分後に、wordの方で、「予想以上に時間のかかる処理です。継続しますか?」との メッセージが表示されます。 「いいえ」を押すと、CloseHandleの行へ進むのですが、 「はい」を押すと、また2~3分後に同じメッセージが表示されます。 やりたい事は、wordに限らず、ファイルに関連付けされたアプリで 起動し、そのファイルが閉じられたら、次の処理に進むということです。 (本来は、これをマルチスレッドで実装予定です) 上記の処理で上手く返ってこないのは、どこか間違った箇所があるのでしょうか? ※ちなみに、WaitForSingleObjectで、INFINITEを指定せず、 WAIT_OBJECT_0が返ってくるまでwhileで回すなどの処理を試してみましたがダメでした。

質問者が選んだベストアンサー

  • ベストアンサー
  • ttega
  • ベストアンサー率52% (9/17)
回答No.1

SHELLEXECUTEINFO を MSDN で調べると書いてありますが、ShellExecuteEx は必ずしも hProcess にプロセスハンドルを入れてくれません。 普通は CreateProcess でプロセスを立ち上げて、そのハンドルで待機します。参考 URL などの例をご覧ください。

参考URL:
http://www.sm.rim.or.jp/~shishido/cprocess.html
lhouse
質問者

お礼

FindExecutable等の処理を省きたかったので CreateProcessを使わなかったのですが、 プロセスハンドルが入るとは限らないのであれば仕方ないですね。 大人しくCreateProcessでコーディングすることにします。

その他の回答 (1)

回答No.2

 こんばんは。  GUIの中でハマッているのでは無いでしょうか。  一応此れで応答しているのですが、駄目でしょうか。  http://msdn.microsoft.com/ja-jp/library/cc429261.aspx SHELLEXECUTEINFO si = {sizeof(si)}; si.fMask = SEE_MASK_NOCLOSEPROCESS; si.hwnd = hDlg; si.lpVerb = _T("open"); si.lpFile = _T("test.doc"); si.nShow = SW_SHOWNORMAL; BOOL rtn; rtn = ShellExecuteEx( &si ); if ( rtn == 0 || (const int)si.hInstApp <= 32 ){ MessageBox(NULL, "open error", "ok", IDOK); return FALSE; } BOOL fQuit = FALSE; MSG msg; while(fQuit == FALSE) { switch(MsgWaitForMultipleObjects(1, &si.hProcess, TRUE, 20, QS_ALLINPUT)) { case WAIT_OBJECT_0 : case WAIT_ABANDONED_0 : fQuit = TRUE; break; case WAIT_TIMEOUT : default : if(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE) == TRUE) { fQuit = (msg.message == WM_QUIT); TranslateMessage(&msg); DispatchMessage(&msg); } } } CloseHandle( si.hProcess ); MessageBox(NULL, "次の処理", "ok", IDOK); break;

lhouse
質問者

お礼

残念ながら、上記コーディングでも 上手くいきませんでした。 今回はCreateProcessでコーディングすることにしました。

関連するQ&A

  • CreateProcessの制御について

    今回exeから別exe(or bat)を起動させる処理を作成しようとしているのですが、 呼び出した後(今回であればnotepad.exe)5秒程度砂時計マークがでる状態になるのです。(バッチでも同じ) なにか原因があるのでしょうか。また、対処方があれば教えてください。 【前提】 ・本体exeにウィンドウは必要なし ・バッチの場合はcmd.exeは非表示 【環境】 ・XP SP2 でコンパイル。 bcc32 5.5.1 と VB6 ・実行はXP SP2 とWin2000 双方で同様の結果 【ソース】 #include<windows.h> int WINAPI WinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance , PSTR lpCmdLine ,int nCmdShow ) { PROCESS_INFORMATION pi; STARTUPINFO si; ZeroMemory(&si,sizeof(si)); si.cb=sizeof(si); CreateProcess(NULL,"notepad.exe",NULL,NULL,FALSE,CREATE_NO_WINDOW|NORMAL_PRIORITY_CLASS, NULL,NULL,&si,&pi); CloseHandle(pi.hThread); WaitForSingleObject(pi.hProcess,INFINITE); CloseHandle(pi.hProcess); return 0; }

  • CreateProcessでのウィンドウサイズの指定方法

    CreateProcessでプログラムを実行するようなプログラムを作成した際に表示されるウィンドウサイズを最大化したり最小化したりするのではなく、ある大きさにしたいのですが、うまくいかないのでご存知の方教えてください。 BOOL testProgram; STARTUPINFO si; PROCESS_INFORMATION pi; memset(&si,0,sizeof(STARTUPINFO)); si.cb = sizeof(STARTUPINFO); si.dwFlags = STARTF_USESHOWWINDOW; si.wShowWindow = SW_SHOWDEFAULT; si.dwXSize = 500;  si.dwYSize = 400; // 例えばこんな風にサイズを指定したいのです testProgram = CreateProcess( NULL, "**.exe", // ここには具体的なものが入ります。 NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi ); if(testProgram!=TRUE)return FALSE; WaitForSingleObject(pi.hProcess,INFINITE); CloseHandle(pi.hProcess); このようなプログラムを書いてみたのですがうまくいきません。(プログラムは抜粋です) 何かいい方法ご存知の方教えていただけないでしょうか? よろしくお願いします。

  • CreateProcess関数について

    ■CreateProcess関数で実行フアイル(.exe)を呼ぶプログラムを作成しています。 ■CreateProcess関数はプロセスを作成すると直に呼出側に戻って来るが、起動したプログラムの実行結果ではない。 ■その様な事を踏まえてプログラム下記にコーテイングしました(概要) ■「呼び出し側」も「呼び出される」もCD-ROM内に有ります。 ■問題はCreateProcess関数で呼んでから、実際に画像が表示されるまでに、時間が掛かる事です。、 ■「質 問」「やりたい事」 CreateProcess関数で呼んでから、実行画面が表示される時間の間に 「その旨のなんだかのメッセージを表示したい」 例えば、Webでのダウンロードやインストールの時の様な... この様な事を、実現するのはどの様にしますか、宜しくお願いします。 STARTUPINFO si; PROCESS_INFORMATION pi; ZeroMemory(&si,sizeof(si)); si.cb=sizeof(si); ZeroMemory(&pi,sizeof(pi)); if(CreateProcess(NULL,(LPTSTR)cmdline,NULL,NULL,FALSE,0,NULL,NULL,&si,&pi)){ CloseHandle(pi.hThread); //CD-ROMから直ぐ表示出来ない場合に備えて //この間で表示に時間を要している事をメッセージ出来ないか? ShowWindow(hWnd,SW_MINIMIZE);//タスクトレイに入れる。 WaitForSingleObject(pi.hProcess,INFINITE); CloseHandle(pi.hProcess); ShowWindow(hWnd,SW_RESTORE););//タスクトレイから出す。 }

  • exeファイルの起動について教えてください

    はじめまして。 現在c言語を勉強しているのですが、プログラム内で別のプログラムのexeファイルを呼び出し起動させたいのですが、サイトをいろいろ回ったのですが上手くいきませんでした。 環境は、Microsoft Visual C++ 2005 Express Edition 現在の呼び出し方法は、 case IDC_AI: /* 表示の次へボタン有効化 */ // SendMessage( hwnd, WM_SETREDRAW, TRUE, 0); /* 表示 */ // ShowWindow( IAhwnd, SW_SHOW); // CGMain(IAhwnd, message, wParam, IParam, nCmdShow2); // ShellExecute(hwnd, "open", "C:\\Documents and Settings\\c319-1037\\デスクトップ\\卒業研究\\プログラム\\11-12\\SQL組と合わせるよう\\Debug\\D3DFWSample2005.exe", NULL, NULL, SW_SHOW); path=(char*)calloc(MAX_PATH+1,sizeof(char)); if(0!=GetModuleFileName( NULL, buf, MAX_PATH )){// 実行ファイルの完全パスを取得 char drive[MAX_PATH+1] ,dir [MAX_PATH+1] ,fname[MAX_PATH+1] ,ext [MAX_PATH+1]; _splitpath(buf,drive,dir,fname,ext);//パス名を構成要素に分解します // file = '"'; // file += "\\"; // file += '"'; file = drive; file += dir; file += "SQL組と合わせるよう\\Debug\\D3DFWSample2005.exe"; // file += '"'; // file += '"'; MessageBox( hwnd, file.c_str(), "完全パス", MB_OK); // MessageBox( hwnd, drive, "ドライブ", MB_OK); // MessageBox( hwnd, dir, "ディレクトリ パス", MB_OK); // MessageBox( hwnd, fname, "ベース ファイル名 (拡張子なし)", MB_OK); } ZeroMemory( &si, sizeof(si) ); si.cb = sizeof(si); ZeroMemory( &pi, sizeof(pi) ); MessageBox( hwnd, "秀丸を起動します", "完全パス", MB_OK); // 秀丸を起動する if( !CreateProcess( NULL, // No module name (use command line). (LPSTR)file.c_str(), // Command line. NULL, // Process handle not inheritable. NULL, // Thread handle not inheritable. FALSE, // Set handle inheritance to FALSE. 0, // No creation flags. NULL, // Use parent's environment block. NULL, // Use parent's starting directory. &si, // Pointer to STARTUPINFO structure. &pi ) // Pointer to PROCESS_INFORMATION structure. ) { MessageBox( hwnd, "CreateProcess failed.", "実行エラー", MB_OK); iRtn = -1; } // Wait until child process exits. WaitForSingleObject( pi.hProcess, INFINITE ); // Close process and thread handles. CloseHandle( pi.hProcess ); CloseHandle( pi.hThread ); MessageBox( hwnd, "秀丸を終了しました", "完全パス", MB_OK); return iRtn; です。 よろしくお願いします。

  • CreateProcessでアプリが終了しない

    ちょっとした変換ツールを作ろうと思い、ダイアログベースの MFCアプリケーションを作成しました。 OKボタンが押された時にCreateProcess()で子プロセスを 起動して、終了したら結果表示して親プロセスも終了する 様にしようと思ったのですが、結果表示までは正常に 実行されるのですが、親プロセスが終了しません。 恐らくCreateProcess()に渡す引数が悪いのだと思うのですが、 どなたかわかる方がいらしたらご教示願います。 void CMscGenSelectorDlg::OnBnClickedOk() {   STARTUPINFO si;   PROCESS_INFORMATION pi;   ZeroMemory( &si, sizeof si );   si.cb = sizeof si;   ZeroMemory( &pi, sizeof pi );   // 子プロセス起動   CreateProcess(     NULL,                // モジュール名     _T("mscgen.exe -Tpng hoge.msc"),  // コマンドライン     NULL,                // セキュリティ記述子     NULL,                // セキュリティ記述子     FALSE,               // ハンドルの継承オプション     0,                 // 作成のフラグ     NULL,                // 新しい環境ブロック     NULL,                // カレントディレクトリ名     &si,                // スタートアップ情報     &pi);                // プロセス情報   // 子プロセス終了待ち   WaitForSingleObject( pi.hProcess, INFINITE );   CloseHandle( pi.hProcess );   CloseHandle( pi.hThread );   // シーケンスチャート表示   ShellExecute( NULL, _T("open"), _T("hoge.png"), NULL, NULL, SW_SHOW );   // 終了   OnOK(); }

  • CreateProcessでEXEを起動させると残像が残る

    お世話になります。 VC++ 6.0 MFC で開発しております。 A.exeから、B.exeを起動しています。 A.exeの(画面上の)上にB.exeが表示されるのですが、 そのB.exeを動かすと白い残像みたいな跡がでます。 そのB.exeの残像みたいなのを出さないようにしたいのですがどのようにすればよいでしょうか ※B.exeを起動している間は、A.exeを操作できないようにしたいのです。 ***********実際のソースです。******** PROCESS_INFORMATION pi; STARTUPINFO si; ZeroMemory(&si,sizeof(si)); si.cb=sizeof(si);   int kekka = CreateProcess(Pass,CommandChar,NULL,NULL,TRUE,NORMAL_PRIORITY_CLASS,NULL,NULL,&si,&pi); if(kekka==0) { CString str; str.Format("起動することはできません。); AfxMessageBox(str, MB_OK, 0); } else { WaitForSingleObject(pi.hProcess,INFINITE); } PeekMessage()とかを使えばいいとか聞いたのですが、まったくわからない状態です。 大変お手数ですが具体的に教えていただければ大変ありがたいです。 なにとぞよろしくお願いします。

  • スレッドについて

    スレッドについて勉強中なのですが、簡単なスレッド作り、 スレッドの処理が終わってからメインの処理を行わせたくて 以下のようなプログラムを書いてみました。 そこで、WaitForSingleObjectを使ってスレッドが終了するのを 待ちたいのですが、WaitForSingleObjectではまってしまうようで sprintfのメッセージ(スレッドが止まってしまっている)が 表示されません。 WRITE関数の処理が終われば、、WaitForSingleObjectで処理が 戻ってくると考えているのですが…。 スレッドについてと、間違いについて教えてもらえないでしょうか? 宜しくお願いします。 main { if(!stop) {   File = CreateFile(...); g_hThead = CreateThread(NULL,0,WRITE,(LPVOID)NULL,0,NULL); g_stop = false; }else { g_stop = true; ::WaitForSingleObject(g_hThead, INFINITE); CloseHandle(File); CloseHandle(hThead ); } } void WRITE() { while(!g_stop) { WriteFile(...); } ...処理を行う sprintf("スレッドを終了します。") }

  • 外部exe呼び出しの方法 ShellExecuteEx, System...

    VC++6での質問す。 ◆やりたいこと◆ (1)内部から外部exeを引数ありで呼び出し、処理をさせます。 (2)その外部exeの結果コードを内部的に受け取り、エラー処理させます。 (3)外部exeの処理実行中は、できればDOS窓を非表示にさせます。 system()呼び出しは最も簡単なのですが、(3)が上手くできませんでした。 ◆やってみた方法◆ こちらの過去質問も参考に見よう見まねでShellExecuteExを使ってやってみようとしました。 ↓ SHELLEXECUTEINFO sei; DWORD dwRC; LPVOID msg; BOOL res; ZeroMemory(&sei, sizeof(SHELLEXECUTEINFO)); sei.cbSize = sizeof(SHELLEXECUTEINFO); sei.nShow = SW_HIDE; sei.lpVerb = (LPCTSTR)cmd; // cmdの中は、"hoge.exe param1 param2" res=ShellExecuteEx( &sei ); WaitForSingleObject( sei.hProcess, INFINITE ); GetExitCodeProcess( sei.hProcess, &dwRC ); CloseHandle(sei.hProcess); if(res != FALSE){ MessageBox( "OK", "result", MB_OK ); }else{ FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dwRC, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &msg, 0, NULL ); MessageBox( msg, "result", MB_OK | MB_ICONERROR ); LocalFree(msg); } 以上で実行しようとすると、 「指定されたファイルに対してこの操作を行うアプリケーションが関連付けられていません・・・」 のエラーになってしまいます。 どのように改善していけばよいでしょうか。 ソースはあちこちのものをミックスして書いていて、半分意味も分からずで、とんでもないことをしているかも知れません(VCは初心者で、ハンドルとかクラス、プロセスとかの知識に乏しいです) また、まったく別で良い方法(CreateProcessの方が簡単だとか)があれば、よろしくお願いします。

  • CreateProcessでうまくexeファイルの起動ができません。

    Win32APIでファイルを読み込むプログラムとCreateProcessによる別に作った、OpenCVの2値化動画像の特徴点抽出をして得られた特徴点の座標を保存するプログラムを起動させるプログラムなのですが、起動はできても特徴点の座標取得ができません。外部起動せずにOpenCVのプログラムを動かすと座標がテキストファイルに保存されます。原因が分からないので、分かる方どなたかご教授願います。 環境はWindowsXP Visual C++ 2008 Express Editionです。 以下がWindowsプログラミングのソースの一部です。 LRESULT CALLBACK WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { STARTUPINFO si; PROCESS_INFORMATION pi; DWORD dwExCode; FILE *fp; errno_t err; TCHAR buf[10]; TCHAR *temp; TCHAR cmdstr[] = _T("C:\\Documents and Settings\\m22015\\My Documents\\Visual Studio 2008\\Projects\\capflow\\Debug\\capflow.exe"); switch(msg){ case WM_CREATE: ZeroMemory(&si, sizeof(si)); ZeroMemory(&pi, sizeof(pi)); si.cb = sizeof(si); if(!CreateProcess(NULL, cmdstr, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS,NULL, NULL, &si, &pi)){ MessageBox(hWnd, TEXT("Can't open capflow.exe"), TEXT("Error"), MB_OK); } while(1){ Sleep(100); GetExitCodeProcess(pi.hProcess, &dwExCode); if(dwExCode == STILL_ACTIVE){ continue; }else{ MessageBox(hWnd, TEXT("Finish"), TEXT("Complete"), MB_OK); break; } } CloseHandle(pi.hProcess); WaitForSingleObject(pi.hProcess, INFINITE); CloseHandle(pi.hThread); break; case WM_KEYUP: switch(LOWORD(wParam)){ case VK_RETURN: err = _tfopen_s(&fp, TEXT("C:\\Documents and Settings\\m22015\\My Documents\\Visual Studio 2008\\Projects\\capflow\\Debug\\Corner.txt"), TEXT("r+")); if(err != 0) break; temp = (TCHAR*)malloc(_tcsclen(buf)+1); if(temp == NULL){ MessageBox(hWnd, TEXT("Can't secure of memory"), TEXT("Error"), MB_OK); break; } while(_fgetts(buf, 10, fp) != NULL){ _tcstok(buf, TEXT("\n")); _tcscpy(temp, buf); MessageBox(hWnd, temp, TEXT("Indication"), MB_OK); } free(temp); fclose(fp); break; } break; case WM_CLOSE: DestroyWindow(hWnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return (DefWindowProc(hWnd, msg, wParam, lParam)); } return 0; }

  • 【C++】セマフォとプロセスの使い方

    【C++】セマフォとプロセスの使い方 WindowsXPでVC++2008コンパイラを使用して、 プロセスとセマフォについて勉強しています。 外部プログラム(今回は電卓として)を5つ立ち上げるのですが、 同時に立ち上げるのは2つまでとしたい場合、どのように組むのでしょうか。 http://nienie.com/~masapico/api_CreateSemaphore.html こちらにスレッドを使用したセマフォを使ったサンプルがあるのでが、 プロセスを使うとなると、うまく書けません。 ご指導頂けませんでしょうか。 ===現在まで書いたソース=== (5つのプロセスをつくるループも実装していませんが、、、) #include<windows.h> #include<stdio.h> #include<tchar.h> HANDLE g_hSemaphre; void main(){ BOOL bRet; STARTUPINFO si; PROCESS_INFORMATION pi; //Semaphoreオブジェクト g_hSemaphre = CreateSemaphore(NULL,2,2,NULL); bRet = CreateProcess(_T("C:\\WINDOWS\\system32\\calc.exe"), _T(""), NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi); //プロセス終了 WaitForSingleObject(pi.hProcess,INFINITE); //後処理(プロセス開放) ReleaseSemaphore(g_hSemaphre,1,NULL); //スレッドとプロセスを閉じる。 CloseHandle(pi.hThread); CloseHandle(pi.hProcess); //Semaphoreオブジェクト CloseHandle(g_hSemaphre); }

専門家に質問してみよう