打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
C#强化系列文章二:在C#中使用钩子

相信以前用过VB、Delphi,特别是VC的程序员应该对钩子程序都不陌生。在C#中我们同样可以使用钩子程序来实现特殊效果,比如当用户按下某个特殊键时提示,比如关闭应用程序前提示等。
当然使用方法相对VC来说要稍微复杂一点,有的地方还不太方便,下面的例子中实现两个基本功能:
1、按下Alt+F4时使窗口最小化
2、关闭应用程序前提示

不过目前只能捕获消息,不能屏蔽消息,我正在实验,也希望知道的高手能多多指教

一、加入winuser.h中的定义
因为钩子程序一般情况下都是在vc下使用的,在c#里面并没有对应的方法、结构等的定义,我们首先需要把winuser.h中的相关定义加入自己的类


    public enum HookType : int
    
{
        WH_JOURNALRECORD 
= 0,
        WH_JOURNALPLAYBACK 
= 1,
        WH_KEYBOARD 
= 2,
        WH_GETMESSAGE 
= 3,
        WH_CALLWNDPROC 
= 4,
        WH_CBT 
= 5,
        WH_SYSMSGFILTER 
= 6,
        WH_MOUSE 
= 7,
        WH_HARDWARE 
= 8,
        WH_DEBUG 
= 9,
        WH_SHELL 
= 10,
        WH_FOREGROUNDIDLE 
= 11,
        WH_CALLWNDPROCRET 
= 12,
        WH_KEYBOARD_LL 
= 13,
        WH_MOUSE_LL 
= 14
    }
具体的说明在msdn中都可以查到,主要的比如WH_KEYBOARD是监控按键事件,WH_CALLWNDPROC是在消息触发时执行


    public enum VirtualKeys
    
{
        VK_SHIFT 
= 0x10,
        VK_CONTROL 
= 0x11,
        VK_MENU 
= 0x12,    //ALT
        VK_PAUSE = 0x13,
        VK_CAPITAL 
= 0x14
    }
这个不用说明了,对应ALT、CTRL等键


    public struct CWPSTRUCT
    
{
        
public IntPtr lparam;
        
public IntPtr wparam;
        
public int message;
        
public IntPtr hwnd;
    }
这个是windows内部传递过来的消息的结构

二、加入自己定义的委托和事件参数

    public delegate int HookProc(int code, IntPtr wParam, IntPtr lParam);
    
public delegate void HookEventHandler(object sender, HookEventArgs e);
HokkProc是SetWindowsHookEx调用时的委托事件,HookEventHandler是自己的委托事件

    public class HookEventArgs : EventArgs
    
{
        
public int HookCode;
        
public IntPtr wParam;
        
public IntPtr lParam;
        
public Keys key;
        
public bool bAltKey;
        
public bool bCtrlKey;
    }
是自己的委托事件中接受的事件参数

三、实现自己的钩子类
这一步是最重要的,要使用钩子,我们需要引用user32.dll中的相应方法:
        [DllImport("user32.dll")]
        
static extern IntPtr SetWindowsHookEx(HookType hook, HookProc callback, IntPtr hMod, uint dwThreadId);

        [DllImport(
"user32.dll")]
        
static extern bool UnhookWindowsHookEx(IntPtr hhk);

        [DllImport(
"user32.dll")]
        
static extern int CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);

        [DllImport(
"user32.dll")]
        
static extern short GetKeyState(VirtualKeys nVirtKey);
SetWindowsHookEx是注册一个钩子程序,UnhookWindowsHookEx是释放钩子程序,CallNextHookEx调用钩子的后续事件处理,GetKeyState得到所按的虚键

然后就可以调用这些方法来实现钩子程序,比如注册一个钩子可以调用:
            m_hook = SetWindowsHookEx(m_hooktype, m_hookproc, IntPtr.Zero, (uint)AppDomain.GetCurrentThreadId());
其中m_hooktype就是HookType中定义的类型,m_hookproc就是实际的钩子处理程序:
m_hookproc = new HookProc(KeyHookProcedure);

最关键的就是KeyHookProcedure等钩子处理程序:
        protected int KeyHookProcedure(int code, IntPtr wParam, IntPtr lParam)
        
{
            
if (code != 0)
            
{
                
return CallNextHookEx(m_hook, code, wParam, lParam);
            }


            
if (HookInvoked != null)
            
{
                Keys key 
= (Keys)wParam.ToInt32();
                HookEventArgs eventArgs 
= new HookEventArgs();
                eventArgs.key 
= key;
                eventArgs.lParam 
= lParam;
                eventArgs.wParam 
= wParam;
                eventArgs.HookCode 
= code;
                eventArgs.bAltKey 
= GetKeyState(VirtualKeys.VK_MENU) <= -127;
                eventArgs.bCtrlKey 
= GetKeyState(VirtualKeys.VK_CONTROL) <= -127;
                HookInvoked(this, eventArgs);
            }


            
return CallNextHookEx(m_hook, code, wParam, lParam);
        }
在这个事件中可以取得消息的参数,特别是按键的值,然后通过HookInvoked委托调用事件实际的处理程序

四、在应用程序中调用钩子类
我们可以在自己的form中声明两个钩子对象
        private MyHook callProcHook = new MyHook(HookType.WH_CALLWNDPROC);
        
private MyHook keyHook = new MyHook(HookType.WH_KEYBOARD);
然后在初始化时注册钩子:
        private void Form1_Load(object sender, EventArgs e)
        
{
            keyHook.HookInvoked 
+= new HookEventHandler(keyHook_HookInvoked);
            keyHook.Install();

            callProcHook.HookInvoked 
+= new HookEventHandler(callProcHook_HookInvoked);
            callProcHook.Install();
        }

然后就是实际的钩子事件:
        private void keyHook_HookInvoked(object sender, HookEventArgs e)
        
{
            
if (e.key == Keys.F4 && e.bAltKey) //Alt + F4
            {
                
this.WindowState = FormWindowState.Minimized;
            }

        }


        
private void callProcHook_HookInvoked(object sender, HookEventArgs e)
        
{
            
unsafe
            
{
                CWPSTRUCT
* message = (CWPSTRUCT*)e.lParam;
                
if (message != null)
                
{
                    
if (message->message == WM_CLOSE)
                    
{
                        (sender 
as MyHook).CallNextProc = false;
                        MessageBox.Show(
"程序即将关闭!");
                    }

                }

            }

        }

这样我们就可以通过钩子实现一些相对底层的应用。

代码说的有点乱,我就把最主要的代码直接列在下面供大家参考:


  1
using System;
  2
using System.Collections.Generic;
  3
using System.ComponentModel;
  4
using System.Data;
  5
using System.Drawing;
  6
using System.Text;
  7
using System.Windows.Forms;
  8
using System.Runtime.InteropServices;
  9

 10
namespace HookTest
 11
{
 12
    
public partial class Form1 : Form
 13
    
{
 14
        
消息定义(WinUser.h中定义)
 20

 21
        
private MyHook callProcHook = new MyHook(HookType.WH_CALLWNDPROC);
 22
        
private MyHook keyHook = new MyHook(HookType.WH_KEYBOARD);
 23

 24
        
public Form1()
 25
        
{
 26
            InitializeComponent();
 27
        }

 28

 29
        
private void Form1_Load(object sender, EventArgs e)
 30
        
{
 31
            keyHook.HookInvoked 
+= new HookEventHandler(keyHook_HookInvoked);
 32
            keyHook.Install();
 33

 34
            callProcHook.HookInvoked 
+= new HookEventHandler(callProcHook_HookInvoked);
 35
            callProcHook.Install();
 36
        }

 37

 38
        
private void keyHook_HookInvoked(object sender, HookEventArgs e)
 39
        
{
 40
            
if (e.key == Keys.F4 && e.bAltKey) //Alt + F4
 41
            {
 42
                
this.WindowState = FormWindowState.Minimized;
 43
            }

 44
        }

 45

 46
        
private void callProcHook_HookInvoked(object sender, HookEventArgs e)
 47
        
{
 48
            
unsafe
 49
            
{
 50
                CWPSTRUCT
* message = (CWPSTRUCT*)e.lParam;
 51
                
if (message != null)
 52
                
{
 53
                    
if (message->message == WM_CLOSE)
 54
                    
{
 55
                        (sender 
as MyHook).CallNextProc = false;
 56
                        MessageBox.Show(
"程序即将关闭!");
 57
                    }

 58
                }

 59
            }

 60
        }

 61

 62
    }

 63

 64
    
消息结构体(参照WinUser.h中定义)
 73

 74
    
钩子类型的枚举
 94

 95
    
虚键值的定义(参照WinUser.h中定义)
105

106
    
钩子委托
110

111
    
钩子事件参数
122

123
    
钩子类
238
}


以上的钩子只对当前应用程序起作用,如果想控制其他的所有程序,需要使用全局钩子。原则上全局钩子在C#中是不支持的,在http://www.codeproject.com/csharp/globalhook.asp 中的代码可以参照来实现全局钩子

本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
C#钩子本线程内消息拦截
C# 区分键盘和红外线扫描枪输入 模拟扫描枪
C#实现鼠标键盘的监控和屏蔽 - 技术应用 - 豆豆网
Hook钩子攻略
Winform 键盘钩子,锁屏小工具的实现(一)
C# 键盘记录器
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服