Компьютерный форум OSzone.net  

Компьютерный форум OSzone.net (http://forum.oszone.net/index.php)
-   Программирование и базы данных (http://forum.oszone.net/forumdisplay.php?f=21)
-   -   Delphi: как закрыть чужую прогу (http://forum.oszone.net/showthread.php?t=30366)

pvr30 19-07-2003 17:15 208363

Возникла необходимость закрыть из моей программы чужое запущенное приложение.
подробнее: надо в определенный момент закрыть чужую программу, как это сделать в WIN XP и Win 9x, какой функцией, запускает например WinExec, а есть ли что то с противоположным эффектм.

vasketsov 19-07-2003 20:33 208364

pvr30
Запусеать надо не через WinExec, а через SHellExecuteEx или CreateProcess. В этих случаях у тебя будет ProcessHandle, и потом его можно передать в функцию TerminateProcess()

mj 19-07-2003 20:48 208365

function  KillProc(const ClassName:AnsiString):Boolean;

 {Terminates the first process with the given window class.  Window class is
  fixed whereas Window title can change.

  Example: KillProc('NOTEPAD') unconditionally terminates Windows Notepad if
           it is running. }
var
 hWnd,hProc:THandle;
 pid:DWORD;
begin
 Result:=False;
 hWnd := FindWindow(PCHAR(ClassName),nil);
 if IsWindow(hWnd) then begin
   GetWindowThreadProcessId(hWnd, @pid);
   hproc := OpenProcess(PROCESS_TERMINATE, FALSE, pid);
   if hproc<>0 then begin
     Result:=TerminateProcess(hProc,0);
     if Result then CloseHandle(hProc);
   end;
 end;
end;

vasketsov 20-07-2003 00:20 208366

mj
Зачем процесс искать и открывать, если процесс создается самолично (следоватьельно, хэндл уже есть). Кроче того, не факт, что свой Notepad найдешь, есть вохможность и чужой прибить. И почему у тебя процесс закрывается, только если удачно прибивается? Закрываться он должен, если открылся.

mj 20-07-2003 10:11 208367

vasketsov
>Зачем процесс искать и открывать, если процесс создается самолично
Откуда ты знаешь что ему надо конкретно, если он этого не написал? Если бы он его создавал сам, то и закрыть было бы не проблема...
>Кроче того, не факт, что свой Notepad найдешь, есть вохможность и чужой прибить
Есть и другие более точные методы поисха handle
>И почему у тебя процесс закрывается, только если удачно прибивается?
чтобы ты спросил... не спорь если не знаешь...
услышал где то, что есть функция TerminateProcess, значит теперь всё можешь ей убивать? а вот и нет, не так всё просто, как кажется...

[s]Исправлено: mj, 13:13 20-07-2003[/s]

pvr30 20-07-2003 15:12 208368

vasketsov
А разве я сказал что эту чужую погу запустил я (про WinExec я ипомянул как сравнение или аналог с обратным эфектом), нет не я. В принцыпе я сделал по другому, я в своей программе использую handle одного из дочерних окон этой проги А1, ну а далее определяю handle A2:=GetWindow (A1,GW_OWNER); родительского окна и делаю SENDMESSAGE (A2,WM_CLOSE,0,0);
Вроде все работает.

Guest 20-07-2003 16:01 208369

Цитата:

SENDMESSAGE (A2,WM_CLOSE,0,0);
Ну это называется вежливо попрость программу закрытся...
Закрывается главное окно и программа завершает работу

vasketsov 21-07-2003 00:37 208370

mj
Цитата:

не спорь если не знаешь...
Давай подискутируем на тему, "необходимо ли вызывать CloseHandle если удачно вызвался OpenProcess". :lol: Идем на msdn.microsoft.com и не путаем остальных.

Итак.

Так как ВСЕ объекты ядра (в том числе процесы) поддерживают подсчет ссылок, то для корректного уничтожения объекта в памяти необходимо для каждого открытия его вызывать фукнцию ZwClose (она вызывается из CloseHandle).

Если процесс создан, то после удачного вызова TerminateProcess он не перестает существовать. В этом легко убедиться, так как после этого хэндл этого процесса продолжает быть валидным, то есть, можно, например, вызвать некоторые классы информации для ZwQueryInformiationProcess (вышележащие функции ее вызывают, например, для получения ExitCode для процесса и времен создания/ работы в UserMode/ KernelMode/ завершения). Если бы объекта ядра "процесс" не существовало, такое получение информации было бы невозможным. Система может узнать, что сам объект можно уничтожить, только если счетчик ссылок на него обнулится. Именно поэтому и надо вызывать CloseHandle независимо от результата TerminateProcess (иначе эта ссылка вычтется только при завершении процесса, который этот хэндл открыл).

Цитата:

Есть и другие более точные методы поисха handle
Ага, особено если и искать не надо. А если есть 2 нотепада, у обоих заголовок - "Untitled-notepad", как ты выберешь, какой прибить, если прибить надо ровно один (например, задача не позволить запускать более одного блокнота)? Необходимо в начальной постановке указывать, например, что он запущен под тем же пользователем, или родитель у него- текущий процесс. Тем более что FindWindow ни при каких обстоятельствах не ищет окна, принадлежащие другой пользовательской сессии (то есть, на другой оконной станции, например, если зашел терминальный юзер или служба выполняется под выделенной учетной записью или системной неинтерактивной), а неименованный десктоп может создать ЛЮБОЙ пользователь в системе, и FindWindow тут обломается найти окно по полной программе. Короче, 2 балла за матчасть.

Цитата:

Если бы он его создавал сам, то и закрыть было бы не проблема...
Проблема, если бы запускал через WinExec или ShellExecute. Они не возвращают хэндл процесса.

Цитата:

услышал где то, что есть функция TerminateProcess, значит теперь всё можешь ей убивать?
Нет. Ей можно завершить процесс, если удастся открыть его с правом TERMINATE и если он не завис на любом системном примитиве синхронизации в ядре. Если процесс работает и есть привилегия отладки, завершить ей можно любой процесс кроме псевдопроцесса SYSTEM (его с указанным правом не открыть) и IDLE (его вообще не открыть, так как у него нулевой идентификатор). А то что кому-то кажется, что там не так все просто - это личные проблемы каждого, а мне все же интересно, что же там может быть такого сложного, что необходимо отказываться от правильной системы подсчета ссылок.

Цитата:

и делаю SENDMESSAGE
В общем случае неверно. Ты такой командой закрываешь окно. А если у программы несколько равнозначных окон (или, по другому, нет главного) - то тебе придется посылать WM_CLOSE им всем.
В зависимости от постановки задачи, может быть можно вызвать GetWindowThreadProcessId и потом OpenProcess/ TerminateProcess/ CloseHandle, а может надо после GetWindowThreadProcessId вызывать EnumWindows и для каждого из них вызывать GetWindowThreadProcessId чтобы каждому не дочернему окну в этом процессе посылать WM_CLOSE. Короче, это уже от задачи зависит. А может будет точнее идентифицировать процесс по имени файла, образ которого он исполняет? Например, не "Untitled-notepad", а notepad.exe?

mj 21-07-2003 11:07 208371

vasketsov
Много слов, а смысла мало...
Пользователь рассматривал конкретный случай, а не писал универсальную программу по убиванию блокнотов с одинаковым именем...

Добавлено:

>Давай подискутируем на тему, "необходимо ли вызывать CloseHandle если удачно вызвался OpenProcess".

Малоли чё где написано, это ведь не я придумал, а Borland, они та моумней тебя... А по поводу того что написано на msdn.microsoft.com: так на заборе тоже много чё пишут...

>Цитата:Если бы он его создавал сам, то и закрыть было бы не проблема...
>Проблема, если бы запускал через WinExec или ShellExecute. Они не возвращают хэндл процесса.

Имелось ввиду сам через CreateProcess

>Цитата:и делаю SENDMESSAGE
>В общем случае неверно.

В любом случае это самый безопасный метод, так что говорить "неверно", просто глупо с твоей стороны...

vasketsov 21-07-2003 15:11 208372

mj
Цитата:

Малоли чё где написано, это ведь не я придумал, а Borland, они та моумней тебя... А по поводу того что написано на msdn.microsoft.com: так на заборе тоже много чё пишут...
То есть, весь мир (включая M$) под M$ пишет неправильно, один борланд правильно? Кстати, откуда этот кусок, что ты на борланд ссылаешься? Попробуй взять кусок кода, открывающий/убивающий процесс/закрывающий его и натравить на процесс, запущенный из Delphi из-под отладчика (в этом случае как рз процесс для системы будет подвешен на примитиве синхронизации типа Event). Возьми HandleEx или аналог с сайта sysinternals от Руссиновича и смотри, как у тебя будет расти количество открытых хэндлов на процесс, открыть ты его сможешь, а закрыть - нет.

Цитата:

В любом случае это самый безопасный метод
Метод можно рассматривать как безопасный или как небезопасный только после того, как он становится рабочим. Если у программы нет понятия главного окна (например, Explorer, запущенный не как Shell и с отключенной опцией создавать для отдельного окна процесс) - ее таким образом не закрыть до закрытия последнего окна либо даже не закрыть совсем (смотря как писать обработчик сообщений и какой критерий завершения программы в этом случае выбрать, просто "нет открытых окон" не всегда корректный критерий для такого случая).

pvr30 21-07-2003 18:15 208373

Да друзья ну и понесло вас.
>Цитата: В общем случае неверно. Ты такой командой закрываешь окно. А если у программы несколько равнозначных окон (или, по другому, нет главного)
А я и не говорил про общий случай и разве я не ясно выразился что, цитата:"В принцыпе я сделал по другому, я в своей программе использую handle одного из ДОЧЕРНИХ окон этой проги А1, ну а далее определяю handle A2:=GetWindow (A1,GW_OWNER); РОДИТЕЛЬСКОГО окна и делаю SENDMESSAGE (A2,WM_CLOSE,0,0); "

>Цитата: Метод можно рассматривать как безопасный или как небезопасный только после того, как он становится рабочим.
Уже работает 2 дня и нормально работает.

[s]Исправлено: pvr30, 20:16 21-07-2003[/s]

RealRascal 27-07-2003 00:28 208374

Здрасьте! А как, глубокоуважаемые, Вы найдете и уничтожите программу у которой нет(допустим, на данный момент нет, или вообще нет) окна? Что если окна в целях экономии ОЗУ создаются, а потом уничтожаются? А вдруг где-нибудь, глубоко в оперативке, сидит себе злобное безоконное приложение и ждет своего часа?

Guest 06-08-2003 08:16 208375

vasketsov

Не подскажешь?

Ситуация: служба создает процесс (пресловутый Notepad),
dwCreationFlags: NORMAL_PRIORITY_CLASS

StartupInfo.dwFlags:=STARTF_USESHOWWINDOW;
StartupInfo.wShowWindow:=SW_SHOW;

Процесс запускается, но без GUI. И при попытке убить процесс с помощью диспетчера задач : "Операция не может быть завершена. Отказано в доступе."

В чем дело? Спасибо.

vasketsov 06-08-2003 19:15 208376

Guest
1) Почему не видно окна. Служба помечена как неинтерактивная, она запускает notepad в рамках системной оконной станции, которая не видна вошедшему пользователю. Для того, чтобы notepad появлялся, надо либо службе разрешить взаимодействие с рабочим столом, либо, если это своя служба и не хочется ее делать интерактивной, научить создавать процесс на конкретном рабочем столе конкретной оконной станции. Необходимая информация есть в msdn в описании функций CreateProcess (и SetProcessWindowStation с SetThreadDesktop, если хочется уметь переключать процесс с одного рабочего стола на другой).

2) Почему не завершается. Для завершения такого процесса необходимо получить привилегии отладки. Как это делается, уже вроде бы писалось, в любом случае искать надо функцию AdjustTokenPrivileges либо здесь, либо в msdn.microsoft.com. Можно еще попробовать использовать RtlAdjustPrivilege, пример находится здесь внизу: http://ntprog.by.ru/_rtl.htm


Время: 17:02.

Время: 17:02.
© OSzone.net 2001-