Python - Windows关机事件

17 投票
2 回答
14324 浏览
提问于 2025-04-15 14:15

当我使用 win32api.setConsoleCtrlHandler() 时,我可以接收到来自Windows的关机、注销等事件,从而可以优雅地关闭我的应用程序。

不过,这个方法只有在通过python.exe运行应用程序时有效(也就是说,它会有一个控制台窗口),而在使用pythonw.exe时就不行了(没有控制台窗口)。

有没有什么办法可以在没有控制台和窗口的情况下,依然接收到这些事件呢?或者,有没有什么编程的方法可以隐藏控制台窗口?

为了更清楚地说明,我的目标是能够在没有任何控制台窗口显示的情况下,成功接收到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)

我觉得我离目标很近了,但肯定还有什么东西遗漏了!

2 个回答

5

如果你没有控制台,那设置控制台处理程序当然是没用的。你可以通过创建另一个窗口(这个窗口不一定要可见)来在图形用户界面(非控制台)程序中接收系统事件。确保这个窗口有一个正常的“消息泵”在运行,并处理 WM_QUERYENDSESSION 消息——这个消息会告诉你的窗口关于关机和注销的事件(你的窗口可以通过对这个消息返回0来尝试阻止会话结束)。另外,“Windows 服务”跟普通应用程序是不同的——如果你在写的是这个,看看这个例子 这里

16

这里的问题是,HWND_MESSAGE这种窗口类型其实收不到广播消息,比如WM_QUERYENDSESSIONWM_ENDSESSION

所以在调用CreateWindowEx()的时候,我没有把win32con.HWND_MESSAGE作为“父窗口”的参数,而是直接用了0

简单来说,这样做会创建一个真正的窗口,但我并不显示这个窗口,所以效果上是一样的。现在,我可以成功接收到那些广播消息,并正确地关闭应用程序。

撰写回答