问题
Linux supports sending an arbitrary Posix-Signal such as SIGINT
or SIGTERM
to a process using the kill
-Command. While SIGINT
and SIGTERM
are just boring old ways to end a process in a friendly or not-so-friendly kind of way, SIGQUIT
is meant to trigger a core dump. This can be used to trigger a running Java VM to print out a thread dump, including the stacktraces of all running threads -- neat! After printing the debugging info, the Java VM will continue doing whatever it was doing before; in fact the thread dump just happens in another spawned thread of maximum priority. (You can try this out yourself by using kill -3 <VM-PID>
.)
Note that you can also register your own signal handlers using the (unsupported!) Signal
and SignalHandler
classes in the sun.misc
-package, so you can have all kinds of fun with it.
However, I have yet to find a way to send a signal to a Windows process. Signals are created by certain user inputs: Ctrl-C
triggers a SIGINT
on both platforms, for instance. But there does not seem to be any utility to manually send a signal to a running, but non-interactive process on Windows. The obvious solution is to use the Cygwin kill
executable, but while it can end Windows processes using the appropriate Windows API, I could not send a SIGBREAK
(the Windows equivalent to SIGQUIT
) with it; in fact I think the only signal it is able to send to Windows processes is SIGTERM
.
So, to make a long story short and to repeat the headline: How to I send an arbitrary signal to a process in Windows?
回答1:
If what you want is to explicitly/programmaticly kill another program/process of any kind, within the SysInternals' pstools there is a small tool named "pskill" that behaves just like Unixen "kill" would do.
If you want something else, keep reading (though I may be wrong on some of the specifics below - it's been eons since I last developed a Windows program in C using only the WinAPI and Charles Petzold's excellent books "Programming for Windows" as a guide).
On Windows you don't properly have "signals", what functions WinMain and WinProc receive from the Operating System are simple messages. For instance, when you click on the "X" button of a window, Windows sends that windows' handler the message WM_CLOSE. When the window's deleted but program's still running, it sends WM_DESTROY. When it's about to get out of the main message processing loop, WinMain (not WinProc) receives WM_QUIT. Your program should respond to all these as expected - you can actually develop an "unclosable" application by not doing what it should upon receiving a WM_CLOSE.
When user selects the task from Windows Task Manager and clicks "End Task", the OS will send WM_CLOSE (and another one I don't remember). If you use "End Process", though, the process is killed directly, no messages sent ever (source: The Old New Thing
I remember there was a way to get the HWND of another process' window, once you get that another process could send that window a message thru functions PostMessage and DispatchMessage.
回答2:
Windows is not POSIX. It does not have signals. The only 'signals' that console programs get is if they call SetConsoleCtrlHandler
, in which case it can be notified that the user has pressed Ctrl+C, Ctrl+Break, closed the console window, logged off, or shut the system down.
Everything else is done with IPC, typically with window messages or RPC. Check Sun's documentation to see if there's a way to do what you're asking on the Windows JRE.
回答3:
In Windows everything revolves around Win32 messages. I do not believe there is a command line tool to do this, but in C++ you could use FindWindow to send an arbitrary message to another Windows program. e.g.:
#define WM_MYMSG ( WM_USER+0x100 )
HWND h = ::FindWindow(NULL,_T("Win32App"));
if (h) {
::PostMessage(h, WM_MYMSG, 0, 0);
}
This can also be done in C# using com interop.
回答4:
You can also use jconsole to view the stacktrace of all the running threads. This will work on Windows, and any other OS that supports Java. jconsole also has many other nice features, memory graphs, cpu graphs, etc.
It doesn't answer your original question, but hopefully allows you to get the same results.
If your not familiar with jconsole, check out the Using JConsole documentation.
回答5:
I'm just wondering if the PsTools from, now Microsoft owned, SysInternals would help you.
回答6:
Ruby is somehow able to (at least emulate) SIGINT SIGKILL etc. on windows, and trap those messages. Might want to check it out.
How ruby does "send signal SIGINT to that process" underneath, in windows, is actually to call TerminateProcess
or equivalent on that PID.
There's also a windows equivalent method for "catching ctrl+c" I imagine it's what it calls there.
来源:https://stackoverflow.com/questions/140111/sending-an-arbitrary-signal-in-windows