首页 程序笔记 .NET C# 使用Hook钩子实现全局监听键盘和鼠标

.NET C# 使用Hook钩子实现全局监听键盘和鼠标

C# 是一种面向对象的编程语言,具有丰富的类库和工具支持,适用于各种类型的应用程序开发。Windows 提供了一种称为"钩子"(Hook)的机制,允许拦截并处理系统级别的事件,如键盘按键和鼠标移动。

通过结合 C# 和 Hook 钩子,我们可以创建一个高效、稳定的全局监听器,捕获所有用户输入,而不仅仅局限于当前应用程序窗口内的活动。

概述

接到客户一个需求,就是在收银员在用扫码枪扫顾客会员码或者微信付款码的时候判断用户有没有加企微好友和进企微群,然后根据这个状态进行语音播报,判断顾客能不能享受优惠价。

关键难点就是用户用的收银系统是别家的,线上小程序用的是我们家的,两家不互通,所以立即决定采用Hook钩子技术做一工具挂在其他收银系统上。

Hook钩子

Windows是一个不断处理消息的系统,每次的鼠标点击移动键盘输入都是消息。钩子是系统消息处理的一环,可以使用钩子处理函数来监听消息传送,并处理消息。

关键技术和工具

为了实现全局监听功能,我们将使用以下技术和工具:

Windows API 钩子:通过调用 Windows API 函数 SetWindowsHookEx 来设置全局钩子。

Low-Level 键盘和鼠标钩子:利用低级钩子 (WH_KEYBOARD_LL 和 WH_MOUSE_LL) 来捕获所有输入事件。

P/Invoke:通过平台调用服务(Platform Invocation Services),从托管代码中调用非托管的 Win32 API 函数。

多线程处理:确保钩子函数不会阻塞主线程,保持应用程序的响应性。

思路

1、用WPF做一个无边框小小的半透明圆形窗体使用TopMost长驻留在桌面上。

2、增加一个ContextMenu 菜单,安装钩子和卸载钩子

3、接入钩子API

示例代码

#region 第一步:声明API函数。
// 安装钩子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern int SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId);
// 卸载钩子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern bool UnhookWindowsHookEx(int idHook);
// 继续下一个钩子
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
public static extern int CallNextHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam);
// 取得当前线程编号
[DllImport("kernel32.dll")]
static extern int GetCurrentThreadId();
#endregion

声明键盘消息结构和鼠标消息结构

public struct KeyMSG
{
    public int vkCode;
    public int scanCode;
    public int flags;
    public int time;
    public int dwExtraInfo;
}
public struct MSG
{
    public Point p;
    public IntPtr HWnd;
    public uint wHitTestCode;
    public int dwExtraInfo;
}

安装键盘钩子和鼠标钩子

#region 第二步:声明,定义委托
public delegate int HookProc(int nCode, Int32 wParam, IntPtr lParam);
static int hKeyboardHook = 0;//如果hKeyboardHook不为0则说明钩子安装成功
HookProc KeyboardHookProcedure;
static int hMouseHook = 0;
HookProc mouseHookProcedure;
#endregion
private void install_Click(object sender, RoutedEventArgs e)
{
    HookStart();
}
private void HookStart()
{
    if (hKeyboardHook == 0)//如果hKeyboardHook==0,钩子安装失败
    {
        //创建HookProc实例
        KeyboardHookProcedure = new HookProc(KeyboardHookProc);
        //设置全局钩子
        hKeyboardHook = SetWindowsHookEx(13, KeyboardHookProcedure,
   Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]), 0);
        if (hKeyboardHook == 0)
        {
            //终止钩子
            throw new Exception("安装钩子失败");
        }
        Trace.WriteLine("键盘钩子安装成功");
    }
    if(hMouseHook == 0)
    {
        mouseHookProcedure = new HookProc(MouseHookProc);
        hMouseHook = SetWindowsHookEx(14, mouseHookProcedure,
   Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]), 0); 
        if(hMouseHook == 0)
        {
            throw new Exception("安装鼠标钩子失败");
        }
        Trace.WriteLine("鼠标钩子安装成功");
    }
}

增加键盘钩子和鼠标钩子处理函数

private StringBuilder codes = new StringBuilder(); //接收扫码枪输入的数据
private int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam)
{
    if (nCode >= 0)
    {
        if(wParam == 0x100)
        {
            KeyMSG m = (KeyMSG)Marshal.PtrToStructure(lParam, typeof(KeyMSG));
            if (m.vkCode == 0x31)
            {
                codes.Append(1);
            }
            else if (m.vkCode == 0x32)
            {
                codes.Append(2);
            }
            else if (m.vkCode == 0x33)
            {
                codes.Append(3);
            }
            else if (m.vkCode == 0x34)
            {
                codes.Append(4);
            }
            else if (m.vkCode == 0x35)
            {
                codes.Append(5);
            }
            else if (m.vkCode == 0x36)
            {
                codes.Append(6);
            }
            else if (m.vkCode == 0x37)
            {
                codes.Append(7);
            }
            else if (m.vkCode == 0x38)
            {
                codes.Append(8);
            }
            else if (m.vkCode == 0x39)
            {
                codes.Append(9);
            }
            else if (m.vkCode == 0x30)
            {
                codes.Append(0);
            }
            else if (m.vkCode == 0x60)
            {
                codes.Append(0);
            }
            else if (m.vkCode == 0x61)
            {
                codes.Append(1);
            }
            else if (m.vkCode == 0x62)
            {
                codes.Append(2);
            }
            else if (m.vkCode == 0x63)
            {
                codes.Append(3);
            }
            else if (m.vkCode == 0x64)
            {
                codes.Append(4);
            }
            else if (m.vkCode == 0x65)
            {
                codes.Append(5);
            }
            else if (m.vkCode == 0x66)
            {
                codes.Append(6);
            }
            else if (m.vkCode == 0x67)
            {
                codes.Append(7);
            }
            else if (m.vkCode == 0x68)
            {
                codes.Append(8);
            }
            else if (m.vkCode == 0x69)
            {
                codes.Append(9);
            }
            if (m.vkCode == 0x0D)
            {
                // 回车键
                string code = codes.ToString();
                Trace.WriteLine(code);
                if(code.Length == 18)
                {
                    StatausHandle(code);
                    codes.Clear();
                }                        
            }
        }                
    }
    return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam);
}

卸载键盘钩子和鼠标钩子

private void unInstall_Click(object sender, RoutedEventArgs e)
{
    HookStop();
}
private void HookStop()
{
    bool retKeyboard = true;
    if (hKeyboardHook != 0)
    {
         retKeyboard = UnhookWindowsHookEx(hKeyboardHook);
         hKeyboardHook = 0;
     }
     if (!retKeyboard)
         throw new Exception("钩子卸载失败");
 
     Trace.WriteLine("键盘钩子卸载成功");
     if (hMouseHook != 0)
     {
         retKeyboard = UnhookWindowsHookEx(hMouseHook);
         hMouseHook = 0;
     }
     if (!retKeyboard)
         throw new Exception("鼠标钩子卸载失败");
     Trace.WriteLine("鼠标钩子卸载成功");
}

好了,拿到的code就算扫码枪输入的条码,根据这个条码实现自己的判断逻辑来进行语音播报就行了。

总结

通过本文的指导,掌握如何使用 C# 实现全局键盘和鼠标监听,从而应用程序增添强大的输入捕获能力。无论是开发自动化工具、辅助技术还是其他需要实时监控用户输入的应用场景,这项技能都将是宝贵的财富。

站心网

C# 是一种面向对象的编程语言,具有丰富的类库和工具支持,适用于各种类型的应用程序开发。Windows 提供了..

为您推荐

.NET C# 过滤从富文本编辑器html里的Javascript脚本

富文本编辑器在允许用户输入丰富内容的同时,也带来了跨站脚本攻击(XSS)的风险。过滤提交的 HTML 中的 <script> 脚本是防止跨站脚本攻击(XSS)的关键步骤。在 .NET C# 服务端过滤 <script> 脚本主要有以下几种方..

亚马逊年销300万美金,“绝美键盘”被老外疯抢!

来源:品牌方舟BrandArk作者:麦林作为出海掘金的热门赛道,3C 行业早已卷成一片血海。既坐拥遥遥领先的规模体量与消费潜力,但也面临着激烈的同质竞争与存量厮杀。但红海之中,也并非所有地带都挤满了人。在部分细..

ZLinq:.NET 高性能 LINQ 替代方案及其使用指南

在 .NET 开发中,LINQ(Language Integrated Query)为数据查询提供了简洁且强大的语法。然而,传统的 LINQ 在处理大量数据时可能会引发性能瓶颈,主要由于频繁的内存分配和对象创建。为解决这一问题,Cysharp 团队..

.NET使用AutoMapper简化对象映射

在.NET软件开发中,常常需要将一个对象的数据转换并映射到另一个对象上。​这种手动映射的过程既繁琐又容易出错,影响开发效率和代码可维护性。​为了解决这一问题,AutoMapper应运而生。​什么是 AutoMapper?AutoM..

.NET C# RESTful API交互Refit库使用教程

Refit 是一个 .NET C# 库,它简化了与 RESTful API 的交互。Refit 受到 Square 的 Retrofit 库的启发,它将 REST API 转换为实时接口,允许你以声明方式定义 REST API 调用。Refit 的特点1. 声明式 API 定义:Refit ..

.NET C# System.Text.Json进阶使用技巧

System.Text.Json 是 .NET 中用于处理 JSON 数据的强大库。除了基本用法外,它还提供了许多进阶技巧,可以帮助你更高效、更灵活地处理 JSON 数据。以下是一些 System.Text.Json 的进阶使用技巧:1. 自定义序列化和反..

ThinkPHP5.0如何全局替换前端某个字符串

在 Nginx 配置文件中添加以下内容:sub_filter'旧字符串''新字符串';sub_filter_onceoff;在.htaccess文件中添加以下内容:<IfModulemod_substitute.c>AddOutputFilterByTypeSUBSTITUTEtext/htmlSubs..

.NET Core 使用ML.NET 机器学习分析预测股票走势

在 .NET Core 中,你可以利用 ML.NET 框架来构建机器学习模型,以预测股票价格走势。以下是一个基本的实现步骤:​1. 准备数据:​收集并整理股票的历史数据,包括日期、开盘价、最高价、最低价、收盘价和成交量等信..

.NET 日志库 Serilog 使用教程

1. Serilog 简介Serilog 是 .NET 生态中强大且灵活的日志库,支持结构化日志记录,并提供多种日志接收器(Sinks),可以将日志输出到控制台、文件、数据库等不同存储介质。Serilog 适用于控制台应用、ASP.NET Core ..

.NET C# 单元测试 mock File.Exists的返回值

在 .NET 单元测试中,使用 Moq 来模拟 File.Exists 方法的返回值,可以这样做:1. 使用 Mock<FileSystem>(推荐).NET 提供了 System.IO.Abstractions 库,你可以使用 Mock<IFileSystem> 来替代 File,这样更符合依..

.NET Core 适配 鸿蒙HarmonyOS 的最新进展

.NET Core适配鸿蒙HarmonyOS的最新进展:运行能力方面目前.Net完全具备可以在OpenHarmony系统上运行的能力。其中,NativeAOT方式是较为可行的一种,它编译出的原生so不依赖glibc,可与鸿蒙系统的libc兼容,能在鸿蒙..

VS创建.NET Core项目使用Docker方式部署到Linux服务器

在 Visual Studio(VS) 中,使用 Docker 方式部署 .NET Core 项目 到 Linux 服务器,可以简化环境管理并提高部署效率。以下是完整教程:1. 在 VS 创建 .NET Core 项目并启用 Docker新建 ASP.NET Core 项目打开 Visu..

.NET C#查询全球IP地址信息 IPTools库 使用教程

IPTools 是一个用于快速查询全球 IP 地址信息的库,支持国内和国际 IP 查询,提供详细的地理位置信息(如国家、省份、城市)以及经纬度等数据。IPTools GitHub地址:https://github.com/stulzq/IPToolsIPTools.China..

2025年.NET 10 和 C# 13 新特性示例

.NET 10预计将于2025年11月正式发布,带来一系列性能优化、开发者效率提升和跨平台能力增强。尽管官方功能集仍在开发中,早期预览版与社区讨论已揭示了多项值得期待的改进。这些增强将基于.NET 9的基础,引入新语言..

2025年做网站还能赚钱吗?

在2025年,互联网的格局虽然不断演变,但建立网站仍然蕴藏着赚钱的潜力。关键在于如何巧妙地定位,以及如何充分利用最新的技术和趋势。首先,我们需要明确网站的类型和盈利模式。内容型网站,如果运营者擅长某一领域..

.NET Core网站减少内存占用的方法

在.NET Core网站开发中,有效管理内存占用对于保证应用程序的性能和稳定性至关重要。以下是一些减少内存占用的关键策略,它们着重于代码优化、内存管理以及相关因素的综合考虑。代码层面的优化首先,编写高效的代码..

.NET平台QR二维码生成库Net.Codecrete.QrCodeGenerator使用教程

今天给大家介绍一个免费的二维码生成库 Net.Codecrete.QrCodeGenerator ,它非常简洁、易用,且可以生成符合二维码标准的二维码图像。Net.Codecrete.QrCodeGenerator是一个开源的二维码生成库,适用于.NET平台。它基..

.NET9 SDK 新版本引入了新的解决方案文件格式.slnx

微软发布了 .NET 9 SDK 的新版本, 在这个新版本中引入了新的解决方案文件格式 slnx 的支持, 现在可以从 dotnet cli 来创建和维护 slnx 了, 并且支持了从 sln 迁移到 slnx, 目前 VisualStudio 和 Rider 都已经支持了 ..

.NET10 C#13最新语法糖用法示例

.NET 10 与 C# 13 带来了更高效的语法和更灵活的类型支持,包括 params 集合扩展、新的 \e 转义序列、方法组的自然类型推断优化,以及 ref struct 允许实现接口等特性,进一步提升了开发体验和代码可读性。C# 13 引..

.NET C# Predicate泛型委托使用方法

Predicate泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由 Array 和 List 类的几种方法使用,用于在集合中搜索元素。 Predicate<T> 通常用于集合的筛选或搜索操作,比如在 List<T> 的 Fi..

发表回复

返回顶部