利用底层键盘钩子屏蔽任意按键
很多人都知道,如果想在系统范围内屏蔽键盘上的任意按键需要使用全局键盘钩子,然而像win键这样“倔强”的按键又不是普通的键盘钩子就能搞定的。这里我提供一种利用底层键盘钩子屏蔽任意按键(包括win键)的方法,并且作成了.dll动态链接库,方便以后使用。钩子,是一种相对复杂一点的技术,通常用来监视系统中某一类型的事件,这些事件可以与某一线程相关(线程钩子),也可以是系统中的所有线程(全局钩子)。关于钩子的理论,我不想说太多,也无法说太多,因为那不是三言两语就能说清楚的。
本文的重点在于底层键盘钩子的应用,前些天CSDN的VB版有人问如何实现屏蔽win键,说实话,这东西用VB也是可以做到的,只不过全局钩子的钩子函数必须写在标准dll中,而VB只能通过变通的方法做出标准dll,稍微有点麻烦,所以我索性用VC写了一个dll,这样VC、VB或Delphi等等都可以调用,而且我也留出了足够的接口,稍后就会看到。
有一点必须得声明一下,底层键盘钩子有一个半致命的缺点,就是只能在NT及其以上系统中使用,不过好在现在用2000、XP、2003的人绝对不在少数,将来用LongHorn的人估计也少不了,所以这点倒是不用太担心。 :)
好了,闲话少说,源代码在此:
DLL头文件(在VC中使用这个DLL中的函数时,需要包含这个头文件,就像使用API要包含windows.h一样):
/********************************************************************/ /* 文件名: MaskKey.h */ /* */ /* 功能: 标准 DLL 导出函数头文件, 在使用该DLL的程序中包含此文件 */ /* */ /* 作者: 卢培培 (goodname008) 时间: 2004.8.21 */ /* */ /* BLOG: http://blog.csdn.net/goodname008 */ /********************************************************************/ DECLSPEC_IMPORT BOOL WINAPI StartMaskKey( LPDWORD lpdwVirtualKey, int nLength, BOOL bDisableKeyboard = FALSE ); DECLSPEC_IMPORT BOOL WINAPI StopMaskKey(); |
DLL主文件:
/********************************************************************/ /* 文件名: MaskKey.cpp */ /* */ /* 功能: 标准 DLL ---- 利用底层键盘钩子实现屏蔽键盘任意按键 */ /* */ /* 作者: 卢培培 (goodname008) 时间: 2004.8.21 */ /* */ /* BLOG: http://blog.csdn.net/goodname008 */ /********************************************************************/ // 导出函数列表 // StartMaskKey // StopMaskKey #define _WIN32_WINNT 0x0500 // 设置系统版本, 确保可以使用底层键盘钩子 #include "windows.h" // 全局变量 LPDWORD g_lpdwVirtualKey = NULL; // Keycode 数组的指针 int g_nLength = 0; // Keycode 数组的大小 BOOL g_bDisableKeyboard = FALSE; // 是否屏蔽整个键盘 HINSTANCE g_hInstance = NULL; // 模块实例句柄 HHOOK g_hHook = NULL; // 钩子句柄 // DLL 入口函数 BOOL APIENTRY DllMain(HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved) { // 保存模块实例句柄 g_hInstance = (HINSTANCE)hModule; // 在进程结束或线程结束时卸载钩子 switch (ul_reason_for_call) { case DLL_PROCESS_ATTACH: break; case DLL_THREAD_ATTACH: break; case DLL_PROCESS_DETACH: case DLL_THREAD_DETACH: delete g_lpdwVirtualKey; if (g_hHook != NULL) UnhookWindowsHookEx(g_hHook); break; } return TRUE; } // 底层键盘钩子函数 LRESULT CALLBACK LowLevelKeyboardProc(int nCode, WPARAM wParam, LPARAM lParam) { // 禁用键盘的某个按键, 如果 g_bDisableKeyboard 为 TRUE 则禁用整个键盘 if (nCode == HC_ACTION) { if (g_bDisableKeyboard) return TRUE; KBDLLHOOKSTRUCT* pStruct = (KBDLLHOOKSTRUCT*)lParam; LPDWORD tmpVirtualKey = g_lpdwVirtualKey; for (int i = 0; i < g_nLength; i++) { if (pStruct->vkCode == *tmpVirtualKey++) return TRUE; } } // 传给系统中的下一个钩子 return CallNextHookEx(g_hHook, nCode, wParam, lParam); } /********************************************************************/ /* 开始屏蔽键盘按键 */ /* */ /* 参数: */ /* lpdwVirtualKey Keycode 数组的指针 */ /* nLength Keycode 数组的大小 */ /* bDisableKeyboard 是否屏蔽整个键盘 */ /* */ /* 返回值: TRUE 成功, FALSE 失败 */ /********************************************************************/ BOOL WINAPI StartMaskKey(LPDWORD lpdwVirtualKey, int nLength, BOOL bDisableKeyboard = FALSE) { // 如果已经安装键盘钩子则返回 FALSE if (g_hHook != NULL) return FALSE; // 将用户传来的 keycode 数组保存在全局变量中 g_lpdwVirtualKey = (LPDWORD)malloc(sizeof(DWORD) * nLength); LPDWORD tmpVirtualKey = g_lpdwVirtualKey; for (int i = 0; i < nLength; i++) { *tmpVirtualKey++ = *lpdwVirtualKey++; } g_nLength = nLength; g_bDisableKeyboard = bDisableKeyboard; // 安装底层键盘钩子 g_hHook = SetWindowsHookEx(WH_KEYBOARD_LL, LowLevelKeyboardProc, g_hInstance, NULL); if (g_hHook == NULL) return FALSE; return TRUE; } /********************************************************************/ /* 停止屏蔽键盘按键 */ /* */ /* 参数: (无) */ /* */ /* 返回值: TRUE 成功, FALSE 失败 */ /********************************************************************/ BOOL WINAPI StopMaskKey() { // 卸载钩子 if (UnhookWindowsHookEx(g_hHook) == 0) return FALSE; g_hHook = NULL; return TRUE; } |
DEF文件(MaskKey.def):
EXPORTS StartMaskKey @1 StopMaskKey @2 |
上面就是DLL工程中主要的三个文件,工程类型为Win32 Dynamic-Link Library。从DEF文件可以看出,DLL共有两个导出函数:StartMaskKey和StopMaskKey。
StartMaskKey有三个参数,lpdwVirtualKey是一个指向DWORD数组的指针,该DWORD数组用来存放virtual-key code,nLength是该数组的大小,bDisableKeyboard是个逻辑值,如果为TRUE表示禁用整个键盘,默认为FALSE。使用正确的参数调用StartMaskKey,DLL可以将DWORD数组中每一个virtual-key code与键盘上对应的按键屏蔽掉,按这些键将完全没有反应(包括win键)。事实上,对于virtual-key code只要一个字节就可以表示了,但KBDLLHOOKSTRUCT结构中的vkCode是DWORD型,所以为求统一我也采用4个字节(DWORD)。纵然如此,微软还是在MSDN中强调了,virtual-key code的值必须是1到254之间的值,这点一定要注意。
StopMaskKey没有参数,表示停止屏蔽键盘按键。如果在程序中没有调用StopMaskKey停止屏蔽键盘按键,在进程或线程退出时将自动停止屏蔽,恢复原来的状态。当然进程和线程一定要正常退出,如果是被别的程序以TerminateProcess或TerminateThread等微软不太建议使用的野蛮手段结束进程或线程的话,就不太好办了。 :(
下面是在VC中调用的例子:(两个Dialog的成员函数,对应两个按钮)
void CMaskKeyAppDlg::OnStartmaskkey() { // 屏蔽 A, B, C, 上, 下, 左, 右及两个win键 DWORD dwVK[] = {'A', 'B', 'C', VK_LEFT, VK_RIGHT, VK_UP, VK_DOWN, VK_LWIN, VK_RWIN}; int nLength = sizeof(dwVK) / sizeof(DWORD); StartMaskKey(dwVK, nLength); } void CMaskKeyAppDlg::OnStopmaskkey() { StopMaskKey(); } |
下面是在VB中调用的例子:(在窗体上添加2个CommandButton,并分别改名为cmdStartMask和cmdStopMask)
Option Explicit Private Declare Function StartMaskKey Lib "MaskKey" (lpdwVirtualKey As Long, ByVal nLength As Long, Optional ByVal bDisableKeyboard As Boolean = False) As Long Private Declare Function StopMaskKey Lib "MaskKey" () As Long Private Sub cmdStartMask_Click() ' 屏蔽 A, B, C, 上, 下, 左, 右及两个win键 Dim key(8) As Long key(0) = vbKeyA key(1) = vbKeyB key(2) = vbKeyC key(3) = vbKeyLeft key(4) = vbKeyRight key(5) = vbKeyUp key(6) = vbKeyDown key(7) = &H5B ' 左边的win键 key(8) = &H 5C ' 右边的win键 StartMaskKey key(0), UBound(key) + 1 End Sub Private Sub cmdStopMask_Click() StopMaskKey End Sub |
对于VB,如果是在VB的IDE环境中按F5启动程序,则必须调用StopMaskKey才能使键盘恢复状态,如果没有调用,则在退出VB的IDE环境时由DLL恢复键盘状态。对于编译后独立执行的VB程序,则和VC编译后的程序一样,无论是否调用StopMaskKey,都将在程序退出时由DLL自动卸载钩子,恢复键盘状态。
其实,钩子并不是什么很深奥的技术,我写这个DLL的目的,也是为了我们在以后用到的时候,可以实行“拿来主义”。
DLL源代码及VC和VB调用例程的下载地址:http://csdngoodname008.51.net/MaskKey.zip
*-------------------------------------------*
* 转载请通知作者并注明出处,CSDN欢迎您! *
* 作者:卢培培(goodname008) *
* 邮箱:goodname008@163.com *
* 专栏:http://blog.csdn.net/goodname008 *
*-------------------------------------------*
利用底层键盘钩子屏蔽任意按键相关推荐
- 利用底层键盘钩子拦载任意按键(回调版)
前段时间我曾经写过一篇<利用底层键盘钩子屏蔽任意按键>,并放到了我的blog上.这篇文章的题目中把"屏蔽"改成了"拦截",显然要比以前的版本强一些了 ...
- HOOK使用:全局键盘钩子
// CatchKey.cpp : Defines the entry point for the DLL application. // #define _WIN32_WINNT 0x0500 // ...
- 钩子原理及实例:实现键盘钩子截获密码
钩子原理及实例:利用鼠标键盘钩子截获密码 钩子原理 钩子能截获系统并得理发送给其它应用程序的消息,能完成一般程序无法完成的功能.Windows系统是建立在事件驱动的机制上的,也就是整个系统都是通过消息 ...
- vs2019键盘钩子_键盘钩子WH_KEYBOARD和WH_KEYBOARD_LL
WH_KEYBOARD和WH_KEYBOARD_LL,这两个都是键盘钩子 区别: 1.MSDN上面说的是WH_KEYBOARD会在应用程序调用GetMessage 或者 PeekMessage函数并且 ...
- VC/MFC 键盘钩子,代码片断
在MFC程序中使用键盘钩子来获取按键消息,例程: #include "StdAfx.h" #include "KeyboardHook.h" static HH ...
- 使用键盘钩子实现挂机锁屏,输入密码解除
使用键盘钩子实现挂机锁屏,输入密码解除 新建.net 4.5 的windows窗体应用程序SystemLockDemo,将默认的form1重命名为FormLaunch. 钩子和windows API类 ...
- 利用键盘钩子开发按键发音程序
前言 一日,看见我妈正在用电脑练习打字,频频低头看键盘,我想:要是键盘能发音的话,不就可以方便她养成"盲打"的好习惯吗?光想不做可不行,开始行动 按键能发音,其关键就是让程序能够知 ...
- C#:键盘钩子的使用,实现键盘屏蔽 及 全局改键功能
1.引入 由于需要使用动态链接库及Process进程类,所以需要在项目中引入命名空间 using System.Diagnostics; using System.Runtime.InteropSer ...
- vs2019键盘钩子_C#键盘按键监视
using System; using System.Diagnostics; using System.IO; using System.Runtime.InteropServices; using ...
最新文章
- python自动华 (十四)
- Linux系统\Centos没有网卡eth0配置文件怎么办?
- python自动化测试脚本可以测php吗_请对比分析一下php的自动化测试与python的自动化测试...
- python自动发邮件附件_python自动发送带附件的邮件(163邮箱,亲测可用)
- 以30字符宽居中输出python字符串_Python3 字符串
- 后缀数组模板 hdu1403
- linux批量创建ftp用户密码,批量在LINUX下建立很多FTP用户
- GBase数据库事务隔离级别
- 自己动手写CPU(6)简单算术操作指令
- X86和X64的区别
- Word多级列表不显示和奇数页显示章节标题问题
- 认识微型计算机 ppt,认识微型计算机(PPT课件).ppt
- VMware vSphere理论
- 怎样使用阿里云国际版ecs实例创建单个云盘快照-Unirech
- CC2530内部结构图
- 测试框架#labview
- OCR在扫描笔中的技术应用
- 0doo-odoo-8快速开发平台pycharm下开发环境搭建
- css 实现四个角加粗
- 科普 | 冬虫夏草为什么那么贵?