Python-Windows关闭事件

2024-06-08 18:34:51 发布

您现在位置:Python中文网/ 问答频道 /正文

使用win32api.setConsoleCtrlHandler()时,我能够从Windows接收关闭/注销/etc事件,并完全关闭我的应用程序。

但是,只有在python.exe下运行应用程序(即,它有一个控制台窗口),而在pythonw.exe下运行应用程序(没有控制台窗口)时,此操作才有效。

当您没有控制台和窗口来接收这些事件时,Windows中是否有等效的方法来接收这些事件?或者,有没有一种编程方法来隐藏控制台窗口?

要说清楚-我的目标是能够成功地接收Windows关闭/注销/etc事件,而不显示任何类型的控制台窗口。

编辑: 我一直在玩,而且我走得更远了。我为此写了一段测试代码。当我执行“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)

我觉得我离这里很近,但我肯定错过了什么!


Tags: importinfolog应用程序windows事件msgexe
2条回答

如果没有控制台,设置控制台处理程序当然无法工作。您可以在一个GUI(非控制台)程序上接收系统事件,方法是使另一个窗口(不必是可见的)、确保它上有一个正常的“消息泵”,并处理WM_QUERYENDSESSION——这是告诉您的窗口关闭和注销事件的消息(您的窗口可以尝试通过返回0来向后推送结束会话对于此消息)。(“Windows服务”与普通应用程序不同——如果这是您正在编写的,请参见示例here)。

这里的问题是HWND_消息窗口类型实际上并不接收广播消息,比如WM_QUERYENDSESSION和WM_ENDSESSION。

因此,我没有为CreateWindowEx()的“父窗口”参数指定win32con.HWND_消息,而是指定了“0”。

基本上,这会创建一个实际的窗口,但我从不显示它,所以它实际上是相同的。现在,我可以成功地接收这些广播消息并正确关闭应用程序。

相关问题 更多 >