Как проверить, работает ли процесс с помощью Delphi?
Подобно этому вопросу, но в Delphi:
Как узнать, запущен ли процесс с помощью С#?
У меня есть программа обновления, я хочу, чтобы она проверила программу, которую она собирается обновить, в настоящее время не работает, желательно, чтобы она проверяла всех пользователей, а не только текущего пользователя.
Ответы
Ответ 1
from: http://www.delphitricks.com/source-code/windows/check_if_a_process_is_running.html
uses TlHelp32;
function processExists(exeFileName: string): Boolean;
var
ContinueLoop: BOOL;
FSnapshotHandle: THandle;
FProcessEntry32: TProcessEntry32;
begin
FSnapshotHandle := CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
FProcessEntry32.dwSize := SizeOf(FProcessEntry32);
ContinueLoop := Process32First(FSnapshotHandle, FProcessEntry32);
Result := False;
while Integer(ContinueLoop) <> 0 do
begin
if ((UpperCase(ExtractFileName(FProcessEntry32.szExeFile)) =
UpperCase(ExeFileName)) or (UpperCase(FProcessEntry32.szExeFile) =
UpperCase(ExeFileName))) then
begin
Result := True;
end;
ContinueLoop := Process32Next(FSnapshotHandle, FProcessEntry32);
end;
CloseHandle(FSnapshotHandle);
end;
procedure TForm1.Button1Click(Sender: TObject);
begin
if processExists('notepad.exe') then
ShowMessage('process is running')
else
ShowMessage('process not running');
end;
Ответ 2
Если вы пишете немного кода автоматического обновления, вы также можете подумать о подключении какого-либо приложения к вашему приложению и сообщить ему, чтобы он закрылся.
Это может быть, например, вовлекайте отправку сообщения в главное окно приложения, в котором говорится, чтобы он закрылся. Или открытие трубы IPC и т.д.
Ответ 3
Если у вас есть контроль над приложением (как это вытекает из вашего вопроса), хорошим способом сделать это является создание именованного объекта сопоставления файлов в начале процесса. Это похоже на предложение о создании мьютекса из RedLEON.
// Add this into the application you wish to update
CreateFileMapping(HWND($FFFFFFFF), nil, PAGE_READONLY, 0, 32, 'MAIN-PROGRAM');
// Note: Mapping object is destroyed when your application exits
// Add this into your updater application
var
hMapping: HWND;
begin
hMapping := CreateFileMapping(HWND($FFFFFFFF), nil, PAGE_READONLY, 0, 32, 'MAIN-PROGRAM');
if (hMapping <> 0) then
begin
if (GetLastError() = ERROR_ALREADY_EXISTS) then
ShowMessage('Application to update is already running!');
end;
Подробнее см. документацию MSDN по CreateFileMapping.
См. также принятый ответ на этот вопрос, который охватывает ответ Люка и предоставляет дополнительные решения.
Ответ 4
Я набираю часть инициализации этих кодов основного блока.
initialization
mHandle := CreateMutex(nil, True, 'myApp.ts');
if GetLastError = ERROR_ALREADY_EXISTS then
begin
MessageDlg('Program already running!', mtError, [mbOK], 0);
Halt;
end;
Ответ 5
uses TlHelp32, PsAPI;
function ProcessExists(anExeFileName: string): Boolean;
var
ContinueLoop: BOOL;
FSnapshotHandle: THandle;
FProcessEntry32: TProcessEntry32;
fullPath: string;
myHandle: THandle;
myPID: DWORD;
begin
// wsyma 2016-04-20 Erkennung, ob ein Prozess in einem bestimmten Pfad schon gestartet wurde.
// Detection wether a process in a certain path is allready started.
// http://stackoverflow.com/info/876224/how-to-check-if-a-process-is-running-using-delphi
// http://swissdelphicenter.ch/en/showcode.php?id=2010
FSnapshotHandle := CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
FProcessEntry32.dwSize := SizeOf(FProcessEntry32);
ContinueLoop := Process32First(FSnapshotHandle, FProcessEntry32);
Result := False;
while Integer(ContinueLoop) <> 0 do
begin
if UpperCase(ExtractFileName(FProcessEntry32.szExeFile)) = UpperCase(ExtractFileName(anExeFileName)) then
begin
myPID := FProcessEntry32.th32ProcessID;
myHandle := OpenProcess(PROCESS_QUERY_INFORMATION or PROCESS_VM_READ, False, myPID);
if myHandle <> 0 then
try
SetLength(fullPath, MAX_PATH);
if GetModuleFileNameEx(myHandle, 0, PChar(fullPath), MAX_PATH) > 0 then
begin
SetLength(fullPath, StrLen(PChar(fullPath)));
if UpperCase(fullPath) = UpperCase(anExeFileName) then
Result := True;
end else
fullPath := '';
finally
CloseHandle(myHandle);
end;
if Result then
Break;
end;
ContinueLoop := Process32Next(FSnapshotHandle, FProcessEntry32);
end;
CloseHandle(FSnapshotHandle);
end;