钩子函数(键盘钩子C#)

本文采用两个案例说明键盘钩子的使用,案例一是使用线程钩子,案例2是系统钩子(实现跨程序获取键值)

 

 

案例一:线程钩子

该例子实现的效果是:拦截键盘的输入,如果输入的是B键或者是b键,就弹出消息框来表示一下拦截到了

步骤:

1 新建Winform应用程序,工程命名为HookDemo,把默认生成的窗体名称改为HookTest

2 新建一个类,名为Hook

编写代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;  //引入的命名空间
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;            //引入的命名空间

namespace HookDemo
{
    internal enum HookType { 
        Keyboard=2   //键盘钩子类型
    }

    //定义委托,该委托的参数是固定这样写
    internal delegate IntPtr HookProc(int code, IntPtr wparam, IntPtr lparam);

    public class Hook
    {
        //记录下一个Hook编号
        static IntPtr _nextHookPtr;

       
        //获取当前线程编号
        [DllImport("kernel32.dll")]
        static extern int GetCurrentThreadId();

        //卸载钩子
        [DllImport("User32.dll")]
        internal extern static void UnhookWindowsHookEx(IntPtr handle);

        //安装钩子
        [DllImport("User32.dll")]
        internal extern static IntPtr SetWindowsHookEx(int idHook,[MarshalAs(UnmanagedType.FunctionPtr)]HookProc Ipfn,IntPtr hanstance,int threadID);


        //获取下一个钩子
        [DllImport("User32.dll")]
        internal extern static IntPtr CallNextHookEx(IntPtr handle,int code,IntPtr wparam,IntPtr Iparam);



        //委托回调的方法
        IntPtr MyHookProc(int code,IntPtr wparam,IntPtr Iparam) {

            if (code < 0)
            {
                //让后面的程序处理该消息
                return CallNextHookEx(_nextHookPtr, code, wparam, Iparam);
            }

            //用户输入的是b或者B
            if (wparam.ToInt32() == 98 || wparam.ToInt32() == 66)
            {
                //设置文本输入框为a
                MessageBox.Show("我拦截到B或者b");
                //该消息结束
                return (IntPtr)1;
            }

            else
            {
                //让后面的程序处理该消息
                return IntPtr.Zero;
            }
            
        }





        /// <summary>
        /// 从外部调用设置钩子
        /// </summary>
        public void SetHook() {
            //已经设置过钩子了,不能重复设置
            if(_nextHookPtr!=IntPtr.Zero){
                return;
            }

            //设置钩子委托回调函数(委托方法注册)
            HookProc myhookProc = new HookProc(MyHookProc);
        
            //把该钩子加到Hook链中
            _nextHookPtr = SetWindowsHookEx((int)HookType.Keyboard,myhookProc,IntPtr.Zero,GetCurrentThreadId());
        }


        /// <summary>
        /// 从外部调用卸载钩子
        /// </summary>
        public void UnHook() { 
            if(_nextHookPtr!=IntPtr.Zero){
                //从Hook链中取消
                UnhookWindowsHookEx(_nextHookPtr);
            }

            _nextHookPtr = IntPtr.Zero;
        }
    }
}

 

3 找到默认生成的窗体,并找到该窗体的FormClosing事件函数,如下图:

4  编写默认窗体下的类HookTest如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace HookDemo
{
    public partial class HookTest : Form
    {
        Hook hook = new Hook();
        public HookTest()
        {
            InitializeComponent();
            //设置钩子
            hook.SetHook();
        }

        
        //页面关闭事件函数
        private void HookTest_FormClosing(object sender, FormClosingEventArgs e)
        {
            //卸载钩子
            hook.UnHook();
        }
    }
}

5 运行程序,并按键盘的B键,输出如下图:

 

 

案例二:系统钩子

 

先看一下例子的效果:

 

拓展:如果把黑窗体隐藏,然后设置为自启动,同时定时判断浏览器输入的url是否为某个url,如果是则进行抓取然后写到自己的数据库,你说能干什么坏事,呵呵,当然这只是想想而已

 

步骤如下:

1 新建Winform应用程序,工程命名为跨程序拦截输入的字符串

2 新建一个类,名为SystemHook

代码如下:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;

namespace 跨程序拦截输入的字符串
{
    class SystemHook
    {
        // 设置钩子
        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        public static extern IntPtr SetWindowsHookEx(int idHook, HookHandlerDelegate lpfn, IntPtr hmod, uint dwThreadID);

        // 卸载钩子 
        [DllImport("user32.dll")]
        public static extern bool UnhookWindowsHookEx(IntPtr idHook);

        // 获取模块句柄
        [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        public static extern IntPtr GetModuleHandle(String modulename);
        public const int WM_KEYDOWN = 0x0100;
        public const int WH_KEYBOARD_LL = 13;
        public const int WM_SYSKEYDOWN = 0x0104;
        public struct KBDLLHOOKSTRUCT
        {
            public int vkCode;
            public int scanCode;
            public int flags;
            public int time;
            public int dwExtraInfo;
        }
        public delegate int HookHandlerDelegate(int nCode, IntPtr wparam, ref KBDLLHOOKSTRUCT lparam);
        //钩子回掉委托实例
        private static HookHandlerDelegate proc;
        //钩子句柄
        private static IntPtr hKeyboardHook;

        private static int HookCallback(int nCode, IntPtr wparam, ref KBDLLHOOKSTRUCT lparam)
        {
            if (
                nCode >= 0
                &&
                (wparam == (IntPtr)WM_KEYDOWN
                ||
                wparam == (IntPtr)WM_SYSKEYDOWN)
                )
            {

                //Console.WriteLine(lparam.vkCode);
                if (lparam.vkCode == 13)
                {
                    Console.WriteLine("按下回车键");
                    return 0;
                }
                else if (lparam.vkCode == 8)
                {
                    Console.WriteLine("按下backspace键");
                    return 0;
                }
                else if (lparam.vkCode == 20)
                {
                    Console.WriteLine("按下大小写切换键");
                    return 0;
                }
                else if (lparam.vkCode ==160||lparam.vkCode==161)
                {
                    Console.WriteLine("按下shift键");
                    return 0;
                }
                else if (lparam.vkCode == 162)
                {
                    Console.WriteLine("按下ctrl键");
                    return 0;
                }
                else if (lparam.vkCode == 190)
                {
                    Console.Write(".");
                    return 0;
                }
                byte[] array = new byte[1];
                array[0] = (byte)(Convert.ToInt32(lparam.vkCode)); //ASCII码强制转换二进制
                Console.Write(Convert.ToString(System.Text.Encoding.UTF8.GetString(array)).ToLower());
            }
            return 0;
        }

        public static void HookStart()
        {
            if (hKeyboardHook == IntPtr.Zero)
            {
                // 创建HookProc实例 
                proc = new HookHandlerDelegate(HookCallback);
                using (Process curPro = Process.GetCurrentProcess())
                using (ProcessModule curMod = curPro.MainModule)
                {
                    //定义全局钩子 
                    hKeyboardHook = SetWindowsHookEx(WH_KEYBOARD_LL, proc, GetModuleHandle(curMod.FileName), 0);
                }

                if (hKeyboardHook == IntPtr.Zero)
                {
                    HookStop();
                    throw new Exception("钩子设置失败");
                }
            }

        }

        public static void HookStop()
        {
            bool retKeyboard = true;
            if (hKeyboardHook != IntPtr.Zero)
            {
                retKeyboard = UnhookWindowsHookEx(hKeyboardHook);
                hKeyboardHook = IntPtr.Zero;
            }
            if (!(retKeyboard)) throw new Exception("卸载钩子失败");

        }
    }
}

3  调用使用,在默认的窗台Form1中输入代码如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace 跨程序拦截输入的字符串
{
    public partial class Form1 : Form
    {
      
        public Form1()
        {
            InitializeComponent();
            this.FormClosing += new System.Windows.Forms.FormClosingEventHandler(this.Form1_FormClosing);
            SystemHook.HookStart();
        }


        private void Form1_FormClosing(object sender, FormClosingEventArgs e)
        {
            SystemHook.HookStop();
        }
    }
}

运行即可得到前面所展示的效果图

  • 3
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
### 回答1: C语言是一种强大的编程语言,具有许多优点。首先,C语言具有高效的执行速度和低延迟。它是一种被广泛应用于嵌入式系统和操作系统开发的语言,因为它可以更好地利用计算机的硬件资源,并提供更快速的执行。 其次,C语言具有简洁而灵活的语法结构,使得开发人员可以方便地编和阅读代码。这使得C语言成为学习编程的理想选择,因为它可以帮助人们理解编程概念和算法的基本原理。 此外,C语言具有广泛的可移植性。C语言编的程序可以在不同的操作系统和平台上运行,包括Windows、Linux和嵌入式系统等。这使得开发人员能够为不同的平台和设备编通用的代码,提高了代码的重用性和可维护性。 另外,C语言还提供了丰富的库函数和工具,可帮助开发人员快速开发和调试程序。这些库函数包括输入输出、字符串操作、数学函数等,提供了很多常用的功能,简化了编程过程。 最后,C语言具有广泛的应用领域。它被用于开发各种应用程序,包括数据库系统、网络和通信系统、游戏开发等。许多流行的编程语言(如C++、Java)都是基于C语言开发的,因此熟练掌握C语言可以为进一步学习其他编程语言打下良好的基础。 总之,C语言的强大之处在于它的高效性、简洁性、可移植性、丰富的库函数和广泛应用领域。这使得C语言成为一种重要的编程语言,并在计算机科学领域具有重要地位。 ### 回答2: C语言是一门功能强大的编程语言。它是一种通用的、高级的计算机编程语言,广泛应用于系统软件和应用软件的开发中。 首先,C语言具有广泛的应用领域。因为C语言是一种高级的编程语言,它可以用来开发各种软件和系统。从操作系统到嵌入式系统,从数据库管理到游戏开发,C语言都能胜任。这使得C语言成为了编程界的一种标准语言。 其次,C语言具有较高的效率和性能。C语言的代码执行效率高,运行速度快。这主要是因为C语言的设计注重底层实现,充分利用了计算机的硬件资源。同时,C语言也提供了丰富的指针操作和内存管理功能,使得程序员能够更精确地控制程序的内存使用,进一步提高了程序的性能。 此外,C语言还具备灵活的语法和丰富的编程库。C语言的语法相对简单,易于学习和使用。它提供了许多基本的数据类型和运算符,同时也支持自定义结构和函数。此外,C语言还拥有庞大的函数库,如标准C库和其他各种第三方库,开发者可以随时调用这些库中的函数来完成各种任务,大大提高了编程的效率。 总之,强大的C语言凭借其广泛应用、高效性能、灵活的语法和丰富的编程库,成为计算机编程中的重要工具。无论是初学者还是专业开发者,掌握C语言都能够让他们在编程领域发挥更强大的能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zxy2847225301

测试使用

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值