C#全局键盘钩子,从控制台应用程序打开一个表单[重复]
Posted
技术标签:
【中文标题】C#全局键盘钩子,从控制台应用程序打开一个表单[重复]【英文标题】:C# global keyboard hook, that opens a form from a console application [duplicate] 【发布时间】:2018-02-11 06:56:12 【问题描述】:所以我有一个带有表单的 C# 控制台应用程序,我想使用热键打开它。例如 Ctrl + 打开表单。所以我现在得到了处理 globalkeylistener 的代码,但看起来我实现它失败了。它做了一个 while 循环来防止它关闭程序,我尝试使用 kbh_OnKeyPressed 方法从用户那里获取输入。
我尝试用这种方式实现它:
using System;
using System.IO;
using System.Collections.Generic;
using System.Linq;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace Globalkey
static class Program
[DllImport("user32.dll")]
private static extern bool RegisterHotkey(int id, uint fsModifiers, uint vk);
private static bool lctrlKeyPressed;
private static bool f1KeyPressed;
[STAThread]
static void Main()
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
LowLevelKeyboardHook kbh = new LowLevelKeyboardHook();
kbh.OnKeyPressed += kbh_OnKeyPressed;
kbh.OnKeyUnpressed += kbh_OnKeyUnpressed;
kbh.HookKeyboard();
while(true)
private static void kbh_OnKeyUnpressed(object sender, Keys e)
if (e == Keys.LControlKey)
lctrlKeyPressed = false;
Console.WriteLine("CTRL unpressed");
else if (e == Keys.F1)
f1KeyPressed = false;
Console.WriteLine("F1 unpressed");
private static void kbh_OnKeyPressed(object sender, Keys e)
if (e == Keys.LControlKey)
lctrlKeyPressed = true;
Console.WriteLine("CTRL pressed");
else if (e == Keys.F1)
f1KeyPressed = true;
Console.WriteLine("F1 pressed");
CheckKeyCombo();
static void CheckKeyCombo()
if (lctrlKeyPressed && f1KeyPressed)
Application.Run(new Form1());
【问题讨论】:
只是忘了说我想在程序运行的时候打开它,所以“ctrl + 只要在keydown事件中捕捉到这个组合键,然后调用表单 它必须是全局的,这意味着我可以在程序不集中时访问热键 那就别再看了,SetWindowsHookEx 来救场了 【参考方案1】:您需要的是一个低级键盘挂钩。
这可能看起来像这样:
using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Windows.Forms;
public class LowLevelKeyboardHook
private const int WH_KEYBOARD_LL = 13;
private const int WM_KEYDOWN = 0x0100;
private const int WM_SYSKEYDOWN = 0x0104;
private const int WM_KEYUP = 0x101;
private const int WM_SYSKEYUP = 0x105;
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc lpfn, IntPtr hMod, uint dwThreadId);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool UnhookWindowsHookEx(IntPtr hhk);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr GetModuleHandle(string lpModuleName);
public delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);
public event EventHandler<Keys> OnKeyPressed;
public event EventHandler<Keys> OnKeyUnpressed;
private LowLevelKeyboardProc _proc;
private IntPtr _hookID = IntPtr.Zero;
public LowLevelKeyboardHook()
_proc = HookCallback;
public void HookKeyboard()
_hookID = SetHook(_proc);
public void UnHookKeyboard()
UnhookWindowsHookEx(_hookID);
private IntPtr SetHook(LowLevelKeyboardProc proc)
using (Process curProcess = Process.GetCurrentProcess())
using (ProcessModule curModule = curProcess.MainModule)
return SetWindowsHookEx(WH_KEYBOARD_LL, proc, GetModuleHandle(curModule.ModuleName), 0);
private IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN || wParam == (IntPtr)WM_SYSKEYDOWN)
int vkCode = Marshal.ReadInt32(lParam);
OnKeyPressed.Invoke(this, ((Keys)vkCode));
else if(nCode >= 0 && wParam == (IntPtr)WM_KEYUP ||wParam == (IntPtr)WM_SYSKEYUP)
int vkCode = Marshal.ReadInt32(lParam);
OnKeyUnpressed.Invoke(this, ((Keys)vkCode));
return CallNextHookEx(_hookID, nCode, wParam, lParam);
要实现它,你可以使用这样的东西:
kbh = new LowLevelKeyboardHook();
kbh.OnKeyPressed += kbh_OnKeyPressed;
kbh.OnKeyUnpressed += kbh_OnKeyUnpressed;
kbh.HookKeyboard();
事件可以这样处理:
bool lctrlKeyPressed;
bool f1KeyPressed;
void kbh_OnKeyPressed(object sender, Keys e)
if (e == Keys.LControlKey)
lctrlKeyPressed = true;
else if (e == Keys.F1)
f1KeyPressed= true;
CheckKeyCombo();
void kbh_OnKeyUnPressed(object sender, Keys e)
if (e == Keys.LControlKey)
lctrlKeyPressed = false;
else if (e == Keys.F1)
f1KeyPressed= false;
void CheckKeyCombo()
if (lctrlKeyPressed && f1KeyPressed)
//Open Form
为了实际理解,我建议您阅读 P/Invoke。那就是利用 Windows 提供的非托管 API。
对于 P/Invoke 可能性的完整列表,pinvoke.net 是一个很好的来源。
为了更好地理解,The official MSDN Website 也是一个很好的来源。
编辑:
看起来您实际上是在使用控制台应用程序,而不是 WinForm 应用程序。在这种情况下,您必须以不同的方式运行程序:
[STAThread]
static void Main()
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
LowLevelKeyboardHook kbh = new LowLevelKeyboardHook();
kbh.OnKeyPressed += kbh_OnKeyPressed;
kbh.OnKeyUnpressed += kbh_OnKeyUnpressed;
kbh.HookKeyboard();
Application.Run();
kbh.UnHookKeyboard();
应用程序类的 Run() 方法为您的应用程序启动一个标准循环。这是 Hook 工作所必需的,因为据我所知,没有此循环的单纯控制台应用程序无法触发这些全局键事件。
使用此实现,按下和释放定义的键会产生以下输出:
注意:我显然替换了
Application.Run(new Form1());
在CheckKeyCombo()
方法中使用
Console.WriteLine("KeyCombo pressed");
【讨论】:
嗯,谢谢,但是如何使用该代码处理多个键输入? 我已经扩展了事件处理代码示例以满足您的需求。请注意,我使用了 CTRL + F1,因为我真的不确定 Keys 枚举中的“msdn.microsoft.com/de-de/library/… 哈哈这不是控制台应用 嗯好吧,我很确定,因为该解决方案适合控制台应用程序:D 道歉! 非常感谢您回答这个问题。链接的明显答案远低于您的答案标准。以上是关于C#全局键盘钩子,从控制台应用程序打开一个表单[重复]的主要内容,如果未能解决你的问题,请参考以下文章