c#键盘钩子全解

c#键盘钩子全解usingSystem;usingSystem.Collections.Generic;usingSystem.Text;usingSystem.Runtime.InteropServices;//调用操作系统动态链接库usingSystem.Reflection;usingSystem.Diagnostics;usingMicrosoft.Win32;usingSys

大家好,又见面了,我是你们的朋友全栈君。
c#键盘钩子全解

using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;//调用操作系统动态链接库
using System.Reflection;
using System.Diagnostics;
using Microsoft.Win32;
using System.Windows.Forms;

namespace util
{
    public class 键盘钩子类
    {
        //使用方法
        //private键盘钩子类hook = new 键盘钩子类();
        //hook.InstallHook(this.拦截函数);
        //public void 拦截函数(键盘钩子类.鼠标信息结构体 hookStruct,out bool handle){};
        //if(hook!=null) hook.UninstallHook();

        //定义常量
        public const int WH_KEYBOARD_LL = 13; //全局钩子键盘为13,线程钩子键盘为2
        public const int WM_KEYDOWN = 0X0100; //键按下0x0101为键弹起
        public static int WM_KEYUP = 0x0101;
        public const int WM_SYSKEYDOWN = 0X0104; 
        //键盘处理事件委托 ,当捕获键盘输入时调用定义该委托的方法.(定义函数指针)
        public delegate int HookHandle(int nCode, int wParam, IntPtr lParam);

        //客户端键盘处理事件
        public delegate void 调用端函数(信息结构体 param, out bool handle);

        //接收SetWindowsHookEx返回值
        private static int 是否以安装 = 0;

        //声明一个指向执行函数的函数指针
        private HookHandle 执行函数引用; 

        //定义存储按键信息的结构体
        [StructLayout(LayoutKind.Sequential)]
        public class 信息结构体
        {
            public int vkCode;
            public int scanCode;
            public int flags;
            public int time;
            public int dwExtraInfo;
        }

        //设置钩子 
        //idHook为13代表键盘钩子为14代表鼠标钩子,lpfn为函数指针,指向需要执行的函数,hInstance为指向进程块的指针,threadId默认为0就可以了
        [DllImport("user32.dll")]
        private static extern int SetWindowsHookEx(int idHook, HookHandle lpfn, IntPtr hInstance, int threadId);

        //取消钩子 
        [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
        private static extern bool UnhookWindowsHookEx(int idHook);

        //调用下一个钩子 
        [DllImport("user32.dll")]
        private static extern int CallNextHookEx(int idHook, int nCode, int wParam, IntPtr lParam);

        //获取当前线程ID(获取进程块)
        [DllImport("kernel32.dll")]
        private static extern int GetCurrentThreadId();

        //Gets the main module for the associated process.
        [DllImport("kernel32.dll")]
        private static extern IntPtr GetModuleHandle(string name);

        private IntPtr 进程块 = IntPtr.Zero;

        //构造器
        public 键盘钩子类() { }

        //外部调用的键盘处理事件
        private static 调用端函数 调用函数引用 = null;

        /// <summary>
        /// 安装勾子
        /// </summary>
        /// <param name="hookProcess">外部调用的键盘处理事件</param>
        public void InstallHook(调用端函数 clientMethod)
        {
            调用函数引用 = clientMethod;

            // 安装键盘钩子 
            if (是否以安装 == 0)
            {
                执行函数引用 = new HookHandle(执行函数);

                进程块 = GetModuleHandle(Process.GetCurrentProcess().MainModule.ModuleName);

                是否以安装 = SetWindowsHookEx(WH_KEYBOARD_LL,执行函数引用,进程块,0);

                //如果设置钩子失败. 
                if (是否以安装 == 0) UninstallHook();
            }
        }

        //取消钩子事件 
        public void UninstallHook()
        {
            if (是否以安装 != 0)
            {
                bool ret = UnhookWindowsHookEx(是否以安装);
                if (ret) 是否以安装 = 0;
            }
        }

        //钩子事件内部调用,调用_clientMethod方法转发到客户端应用。
        private static int 执行函数(int nCode, int wParam, IntPtr lParam)
        {
            if (nCode >= 0 && (wParam == (int)WM_SYSKEYDOWN) || (wParam == (int)WM_KEYDOWN))
            {
                //转换结构
                信息结构体 hookStruct = (信息结构体)Marshal.PtrToStructure(lParam, typeof(信息结构体));

                if (调用函数引用 != null)
                {
                    bool handle = false;   //默认不拦截
                    //调用客户提供的事件处理程序。
                    调用函数引用(hookStruct, out handle);
                    if (handle) return 1; //1:表示拦截键盘,return 退出
                }
            }
            return CallNextHookEx(是否以安装, nCode, wParam, lParam);
        }

    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/130999.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • cmd切换盘符命令

    cmd切换盘符命令cmd切换盘符

    2022年10月4日
    0
  • 软件稳定性测试_兼容性测试

    软件稳定性测试_兼容性测试1 稳定性测试就测试系统的长期稳定运行能力。在系统运行过程中,对系统施压,观察系统的各种性能指标,以及服务器的指标。2 测试场景:模拟平常的压力,模拟实际中日常的用户数进行操作。数据库要存有一定的数据。3 稳定性测试是概率性的测试,就是说即使稳定性测试通过,也不能保证系统实际运行的时候不出问题。所以要尽可能的提高测试的可靠性。可以通过多次测试,延长测试时间,增大测试压力来提高测试的

    2022年9月10日
    0
  • 项目管理知识体系五大过程组(项目管理10大知识领域)

    信息系统项目管理师属于计算机技术与软件专业技术资格(水平)考试五大高级资格考试中的一项,是信息系统建设领域的高级项目经理资质申报的直接性前提条件。通过本考试的合格人员能够掌握信息系统项目管理的知识体系,具有高级工程师的实际工作能力和业务水平。通过本考试的基本用途有以下2个:1、评职称:此证书在国企和事业单位可以评职称。2、申报项目经理:此证书可在自己所在企业申报系统集成项目经理,如果…

    2022年4月11日
    96
  • PyCharm激活码永久有效PyCharm2017.3.5激活码教程-持续更新,一步到位「建议收藏」

    PyCharm激活码永久有效PyCharm2017.3.5激活码教程-持续更新,一步到位「建议收藏」PyCharm激活码永久有效2017.3.5激活码教程-Windows版永久激活-持续更新,Idea激活码2017.3.5成功激活

    2022年6月19日
    24
  • java volatile原理

    java volatile原理一、基本概念先补充一下概念:Java内存模型中的可见性、原子性和有序性。可见性:  可见性是一种复杂的属性,因为可见性中的错误总是会违背我们的直觉。通常,我们无法确保执行读操作的线程能适时地看到其他线程写入的值,有时甚至是根本不可能的事情。为了确保多个线程之间对内存写入操作的可见性,必须使用同步机制。  可见性,是指线程之间的可见性,一个线程修改的状态对另一个线程是可见的。也就是…

    2022年7月18日
    10
  • Nginx+PHP(laravel) 环境 499 错误码排查过程小记

    Nginx+PHP(laravel) 环境 499 错误码排查过程小记

    2022年2月17日
    41

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号