使用win32api.setConsoleCtrlHandler()时,我能够从Windows接收关闭/注销/ etc事件,并干净地关闭我的应用程序。
但是,这仅在python.exe下运行应用程序时有效(即,它有一个控制台窗口),但不在pythonw.exe下(没有控制台窗口)。
当您没有控制台且没有窗口可以接收这些事件时,Windows中是否有相同的方式来接收这些事件?或者,是否有一种隐藏控制台窗口的编程方式?
要明确 - 我的目标是能够成功接收Windows关闭/注销/等事件,而不会显示任何类型的控制台窗口。
修改 我一直在玩,而且我已经走得更远了。我为此写了一段测试代码。当我执行“taskkill / im pythonw.exe”时 - 它将收到消息。
但是,当我在Windows上执行关机,重启或注销时,我没有收到任何消息。
以下是整个事情:
""" Testing Windows shutdown events """
import win32con
import win32api
import win32gui
import sys
import time
def log_info(msg):
""" Prints """
print msg
f = open("c:\\test.log", "a")
f.write(msg + "\n")
f.close()
def wndproc(hwnd, msg, wparam, lparam):
log_info("wndproc: %s" % msg)
if __name__ == "__main__":
log_info("*** STARTING ***")
hinst = win32api.GetModuleHandle(None)
wndclass = win32gui.WNDCLASS()
wndclass.hInstance = hinst
wndclass.lpszClassName = "testWindowClass"
messageMap = { win32con.WM_QUERYENDSESSION : wndproc,
win32con.WM_ENDSESSION : wndproc,
win32con.WM_QUIT : wndproc,
win32con.WM_DESTROY : wndproc,
win32con.WM_CLOSE : wndproc }
wndclass.lpfnWndProc = messageMap
try:
myWindowClass = win32gui.RegisterClass(wndclass)
hwnd = win32gui.CreateWindowEx(win32con.WS_EX_LEFT,
myWindowClass,
"testMsgWindow",
0,
0,
0,
win32con.CW_USEDEFAULT,
win32con.CW_USEDEFAULT,
win32con.HWND_MESSAGE,
0,
hinst,
None)
except Exception, e:
log_info("Exception: %s" % str(e))
if hwnd is None:
log_info("hwnd is none!")
else:
log_info("hwnd: %s" % hwnd)
while True:
win32gui.PumpWaitingMessages()
time.sleep(1)
我觉得我离这儿很近,但我肯定错过了什么!
答案 0 :(得分:9)
这里的问题是HWND_MESSAGE窗口类型实际上不接收广播消息 - 如WM_QUERYENDSESSION和WM_ENDSESSION。
因此,我没有为CreateWindowEx()的“父窗口”参数指定win32con.HWND_MESSAGE,而是指定了“0”。
基本上,这会创建一个实际的窗口,但我从不显示它,所以它实际上是相同的。现在,我可以成功接收这些广播消息并正确关闭应用程序。
答案 1 :(得分:4)
如果您没有控制台,设置控制台处理程序当然无法正常工作。您可以通过创建另一个窗口(不一定是可见的)在GUI(非控制台)程序上接收系统事件,确保在其上有正常的“消息泵”服务,并处理WM_QUERYENDSESSION
- - 这是告诉窗口有关关闭和注销事件的消息(并且您的窗口可以尝试通过为此消息返回0来反击结束会话)。 (“Windows服务”与普通应用程序不同 - 如果您正在撰写,请参阅示例here)。