在之前寫的一篇[《Python:監控鍵盤輸入、鼠標操作,并將捕獲到的信息記錄到文件中》](http://blog.csdn.net/dyx1024/article/details/7311013)文章中,有個讀者留言如下:

這看似一個很平常的需求,但實現起來并不容易,如果用快捷鍵來控制一個程序干些別的事情那是非常容易的,但關鍵是本程序剛好是用hook來監控鍵盤,所以必須使用PumpMessages(),而此函數使用當前程序進入消息循環,它抓取每個鼠標和鍵盤事件。當我們的程序跑起來后,按下停止的熱鍵時,也被此函數捕獲,所以定義的任何熱鍵均不能生效,具體實現及測試在文章[《Python:通過自定義系統級快捷鍵來控制程序運行》](http://blog.csdn.net/dyx1024/article/details/7335085)中有所描述。
現在,我們換一個思路,既然已經監控到了按鍵,那就判斷當前的按鍵是不是預先定義的熱鍵,如果是,則可調用自己的處理函數,這樣就找到了一個控制的入口,可以通過它實現我們想要的功能,注意此時不能讓程序調用os.exit(0)讓程序退出,否則再次按啟動熱鍵時就沒法玩了。具體實現如下,代碼中有詳細注釋,不再一一解釋。
### 一、代碼:
~~~
#!/usr/bin/env python
# -*- coding: utf-8 -*-
import pythoncom
import pyHook
import time
import pyhk
import os
import sys
import ctypes
from ctypes import wintypes
import win32con
import win32api
class CInspectKeyAndMouseEvent:
'''
Function:鍵盤和鼠標監控類
Input:NONE
Output: NONE
author: socrates
blog:http://blog.csdn.net/dyx1024
date:2012-03-09
'''
def __init__(self, filename):
'初始化'
self.filename = filename
def open_file(self):
'打開文件'
self.fobj = open(self.filename, 'w')
def close_file(self):
'關閉文件'
self.fobj.close()
def IsNotWriteLog(self):
'是否記錄日志'
return self.bFlag
def IsExitCommand(self, event):
'''
是否當前按下了程序定義的熱鍵'
如果按下了ALT+F2,將記錄日志的狀態位置為True,不記錄日志,
如果按下了ALT+F1,將記錄日志狀態位置為False,表示記錄日志
'''
if event.Alt == 32 and str(event.Key) == 'F2':
self.bFlag = True
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' stop write log'
elif event.Alt == 32 and str(event.Key) == 'F1':
self.bFlag = False
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' start write log'
def onMouseEvent(self, event):
"處理鼠標事件"
#判斷是否要記錄日志
if self.IsNotWriteLog():
return True
self.fobj.writelines('-' * 20 + 'MouseEvent Begin' + '-' * 20 + '\n')
self.fobj.writelines("Current Time:%s\n" % time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time())))
self.fobj.writelines("MessageName:%s\n" % str(event.MessageName))
self.fobj.writelines("Message:%d\n" % event.Message)
self.fobj.writelines("Time_sec:%d\n" % event.Time)
self.fobj.writelines("Window:%s\n" % str(event.Window))
self.fobj.writelines("WindowName:%s\n" % str(event.WindowName))
self.fobj.writelines("Position:%s\n" % str(event.Position))
self.fobj.writelines('-' * 20 + 'MouseEvent End' + '-' * 20 + '\n')
return True
def onKeyboardEvent(self, event):
#處理按下的熱鍵
self.IsExitCommand(event)
#判斷是否要記錄日志
if self.IsNotWriteLog():
return True
self.fobj.writelines('-' * 20 + 'Keyboard Begin' + '-' * 20 + '\n')
self.fobj.writelines("Current Time:%s\n" % time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time())))
self.fobj.writelines("MessageName:%s\n" % str(event.MessageName))
self.fobj.writelines("Message:%d\n" % event.Message)
self.fobj.writelines("Time:%d\n" % event.Time)
self.fobj.writelines("Window:%s\n" % str(event.Window))
self.fobj.writelines("WindowName:%s\n" % str(event.WindowName))
self.fobj.writelines("Ascii_code: %d\n" % event.Ascii)
self.fobj.writelines("Ascii_char:%s\n" % chr(event.Ascii))
self.fobj.writelines("Key:%s\n" % str(event.Key))
self.fobj.writelines('-' * 20 + 'Keyboard End' + '-' * 20 + '\n')
return True
#默認記錄
bFlag = False
def InspectKeyAndMouseEvent():
"啟動監控"
my_event = CInspectKeyAndMouseEvent("D:\\hook_log.txt")
my_event.open_file()
#創建hook句柄
hm = pyHook.HookManager()
#監控鍵盤
hm.KeyDown = my_event.onKeyboardEvent
hm.HookKeyboard()
#監控鼠標
hm.MouseAll = my_event.onMouseEvent
hm.HookMouse()
#循環獲取消息
pythoncom.PumpMessages()
my_event.close_file()
def handle_start_InspecEvent():
"開始監控(按下Ctrl + F1)"
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' start write log'
InspectKeyAndMouseEvent()
#def handle_stop_InspecEvent():
# "停止監控 (按下Ctrl + F2)"
# InspectKeyAndMouseEvent(False)
if __name__ == "__main__":
'''
Function:通過快捷鍵控制程序運行
Input:NONE
Output: NONE
author: socrates
blog:http://blog.csdn.net/dyx1024
date:2012-03-09
'''
byref = ctypes.byref
user32 = ctypes.windll.user32
#定義快捷鍵
HOTKEYS = {
1 : (win32con.VK_F1, win32con.MOD_ALT)
# 2 : (win32con.VK_F2, win32con.MOD_ALT)
}
#快捷鍵對應的驅動函數
HOTKEY_ACTIONS = {
1 : handle_start_InspecEvent,
# 2 : handle_stop_InspecEvent
}
#注冊快捷鍵
for id, (vk, modifiers) in HOTKEYS.items ():
if not user32.RegisterHotKey (None, id, modifiers, vk):
print "Unable to register id", id
#啟動監聽
try:
msg = wintypes.MSG ()
while user32.GetMessageA (byref (msg), None, 0, 0) != 0:
if msg.message == win32con.WM_HOTKEY:
action_to_take = HOTKEY_ACTIONS.get (msg.wParam)
if action_to_take:
action_to_take ()
user32.TranslateMessage (byref (msg))
user32.DispatchMessageA (byref (msg))
finally:
for id in HOTKEYS.keys ():
user32.UnregisterHotKey (None, id)
~~~
### 二、測試:
1、以下打印是按下熱鍵時控制臺輸出(支持當前程序不是非激活窗口下按下熱鍵)

2、日志內容:
可以看到,在23:16:58停止記錄日志后,至23:17:05重新開始記錄之前,所有的鍵盤和鼠標輸入均沒有記錄,達到預期效果。
~~~
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:16:57]:?
MessageName:mouse move
Message:512
Time_sec:12542031
Window:328916
WindowName:FolderView
Position:(737, 438)
--------------------MouseEvent End--------------------
--------------------Keyboard Begin--------------------
Current Time:[2012-03-09 23:16:58]:?
MessageName:key sys down
Message:260
Time:12542890
Window:1639322
WindowName:本地磁盤 (D:)
Ascii_code: 0
Ascii_char:?
Key:Lmenu
--------------------Keyboard End--------------------
--------------------Keyboard Begin--------------------
Current Time:[2012-03-09 23:17:05]:?
MessageName:key sys down
Message:260
Time:12550015
Window:1639322
WindowName:本地磁盤 (D:)
Ascii_code: 0
Ascii_char:?
Key:F1
--------------------Keyboard End--------------------
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:17:06]:?
MessageName:mouse move
Message:512
Time_sec:12551000
Window:328916
WindowName:FolderView
Position:(720, 420)
--------------------MouseEvent End--------------------
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:17:06]:?
MessageName:mouse move
Message:512
Time_sec:12551015
Window:328916
WindowName:FolderView
Position:(719, 420)
--------------------MouseEvent End--------------------
~~~
- 前言
- Python:實現文件歸檔
- Pyhon:按行輸出文件內容
- Python:讀文件和寫文件
- Python:實現一個小算法
- Python:通過命令行發送新浪微博
- Python:通過攝像頭實現的監控功能
- Python:通過攝像頭抓取圖像并自動上傳至新浪微博
- Python:簡單的攝像頭程序實現
- Python:日志模塊logging的應用
- Python:操作嵌入式數據庫SQLite
- Python:將句子中的單詞全部倒排過來,但單詞的字母順序不變
- Python:語音處理,實現在線朗讀RFC文檔或本地文本文件
- Python:通過計算階乘來學習lambda和reduce這兩個函數的使用
- Python:通過執行100萬次打印來比較C和python的性能,以及用C和python結合來解決性能問題的方法
- Python:使用matplotlib繪制圖表
- Python:使用pycha快速繪制辦公常用圖(餅圖、垂直直方圖、水平直方圖、散點圖等七種圖形)
- Python:使用pycha快速繪制辦公常用圖二(使用樣式定制個性化圖表)
- Python:監控鍵盤輸入、鼠標操作,并將捕獲到的信息記錄到文件中
- Python:通過獲取淘寶賬號和密碼的實驗,來看登陸方式選擇的重要性
- Python:通過獲取淘寶賬號和密碼的實驗,來看登陸方式選擇的重要性(二)
- Python:通過遠程監控用戶輸入來獲取淘寶賬號和密碼的實驗(一)
- Python:通過遠程監控用戶輸入來獲取淘寶賬號和密碼的實驗(二)
- Python:通過自定義系統級快捷鍵來控制程序運行
- Python:通過自定義系統級快捷鍵來控制程序開始或停止記錄日志(使用小技巧解決一個貌似無解的問題)
- Python:一個多功能的抓圖工具開發(附源碼)
- Python:程序發布方式簡介一(打包為可執行文件EXE)
- Python:新浪微博應用開發簡介(認證及授權部分)
- Python:程序最小化到托盤功能實現
- Python:實用抓圖工具開發介紹(含需求分析、設計、編碼、單元測試、打包、系統測試、發布各環節)
- Python:桌面氣泡提示功能實現
- Python:未來三個月的python學習計劃
- Python:pygame模塊及SDL庫簡介
- Python:獲取新浪微博用戶的收聽列表和粉絲列表
- Python:pygame游戲編程之旅一(Hello World)
- Python:pygame游戲編程之旅二(自由移動的小球)
- Python:pygame游戲編程之旅三(玩家控制的小球)
- Python:pygame游戲編程之旅四(游戲界面文字處理)
- Python:pygame游戲編程之旅五(游戲界面文字處理詳解)
- Python:pygame游戲編程之旅六(游戲中的聲音處理)
- Python:pygame游戲編程之旅七(pygame基礎知識講解1)
- Python:編程“八榮八恥”之我見
- Python:腳本的幾種執行方式
- wxPython:簡單的wxPython程序
- wxPython:簡單的wxPython程序的另一種寫法
- wxPython:應用程序對象介紹
- wxPython:輸出重定向
- wxPython:關閉wxPython程序
- wxPython:Frame類介紹
- wxPython:面板Panel的使用
- wxPython:工具欄、狀態欄、菜單實現
- wxPython:消息對話框MessageDialog
- wxPython:文本對話框TextEntryDialog
- wxPython:列表選擇框SingleChoiceDialog
- wxPython:事件處理介紹一
- wxPython:事件處理介紹二
- wxPython: 簡單的繪圖例子
- wxPython:狀態欄介紹
- wxPython:菜單介紹
- wxPython:文件對話框wx.FileDialog
- wxPython:顏色選擇對話框wx.ColourDialog
- wxPython:布局管理器sizer介紹
- wxPython:啟動畫面SplashScreen介紹
- wxPython:繪畫按鈕BitmapButton介紹
- wxPython:進度條Gauge介紹
- Python: 發送新浪微博(使用oauth2)
- Python:讀取新浪微博收聽列表
- Python:DNS客戶端實現