zoukankan      html  css  js  c++  java
  • 让你的上位机程序独占鳌头

    前言

    有学员提出要求,上位机程序启动之后,可以禁用Win组合,防止操作人员无操作或者退出程序。

    实现思路

    首先我们要了解一下键盘在PC上的工作原理,Windows系统所有的操作都是基于消息机制的,也就是说我们键盘上的每个按键其实都是往Windows底层发送一个消息,所以如果想要屏蔽掉某个按键或者功能,最直接的办法就是消息拦截。

    再说一下实现思路,就是使用钩子Hook,这个我们之前讲扫码枪的时候也讲过一点。

    钩子原理

    钩子是操作系统消息处理的一种机制。通过钩子,应用程序可以安装一个钩子回调过程让系统调用,从而监视系统中的消息队列。在这些消息到达目标窗口之前对这些消息进行处理。

    钩子特点

    • 钩子函数会降低操作系统的性能,因为它增加系统处理每一个消息的开销。

    • 操作系统支持多种类型的钩子,每种类型都提供了它特有的消息处理机制。

    • 对于每种类型的钩子,系统都维护一个各自独立的钩子链,钩子链是一个指向用户提供的回调函数钩子过程的链表指针。

    • 钩子尽量避免大量使用,对于一个钩子,一般是需要的时候安装,使用完成后,尽快卸载。

    代码实现

    下面就直接贴代码了,主要是在键盘钩子处理那里加了一些需要截获的消息。

       public class Hook : IDisposable
        {
            public delegate int HookProc(int nCode, int wParam, IntPtr lParam);
            static int hHook = 0;
            public const int WH_KEYBOARD_LL = 13;
            HookProc KeyBoardHookProcedure;
            [StructLayout(LayoutKind.Sequential)]
            public class KeyBoardHookStruct
            {
                public int vkCode;
                public int scanCode;
                public int flags;
                public int time;
                public int dwExtraInfo;
            }
            [DllImport("user32.dll")]
            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")]
            public static extern int CallNextHookEx(int idHook, int nCode, int wParam, IntPtr lParam);
            [DllImport("kernel32.dll")]
            public static extern IntPtr GetModuleHandle(string name);
            public void Start()
            {
                // 安装键盘钩子 
                if (hHook == 0)
                {
                    KeyBoardHookProcedure = new HookProc(KeyBoardHookProc);
                    hHook = SetWindowsHookEx(WH_KEYBOARD_LL, KeyBoardHookProcedure, GetModuleHandle(Process.GetCurrentProcess().MainModule.ModuleName), 0);
                    //如果设置钩子失败. 
                    if (hHook == 0)
                        Close();
                    else
                    {
                        RegistryKey key = Registry.CurrentUser.OpenSubKey(@"Software\Microsoft\Windows\CurrentVersion\Policies\System", true);
                        if (key == null)//如果该项不存在的话,则创建该项
                            key = Registry.CurrentUser.CreateSubKey(@"Software\Microsoft\Windows\CurrentVersion\Policies\System");
                        key.SetValue("DisableTaskMgr", 1, RegistryValueKind.DWord);
                        key.Close();
                    }
                }
            }
            public void Close()
            {
                bool retKeyboard = true;
                if (hHook != 0)
                {
                    retKeyboard = UnhookWindowsHookEx(hHook);
                    hHook = 0;
                }
                RegistryKey key = Registry.CurrentUser.OpenSubKey(@"Software\Microsoft\Windows\CurrentVersion\Policies\System", true);
                if (key != null)
                {
                    key.DeleteValue("DisableTaskMgr", false);
                    key.Close();
                }
            }
            public static int KeyBoardHookProc(int nCode, int wParam, IntPtr lParam)
            {
                if (nCode >= 0)
                {
                    KeyBoardHookStruct kbh = (KeyBoardHookStruct)Marshal.PtrToStructure(lParam, typeof(KeyBoardHookStruct));
                    if (kbh.vkCode == 91) // 截获左win(开始菜单键) 
                        return 1;
                    if (kbh.vkCode == 92)// 截获右win 
                        return 1;
                    if (kbh.vkCode == (int)Keys.Escape && (int)Control.ModifierKeys == (int)Keys.Control) //截获Ctrl+Esc 
                        return 1;
                    if (kbh.vkCode == (int)Keys.F4 && (int)Control.ModifierKeys == (int)Keys.Alt) //截获alt+f4 
                        return 1;
                    if (kbh.vkCode == (int)Keys.Tab && (int)Control.ModifierKeys == (int)Keys.Alt) //截获alt+tab 
                        return 1;
                    if (kbh.vkCode == (int)Keys.Escape && (int)Control.ModifierKeys == (int)Keys.Control + (int)Keys.Shift) //截获Ctrl+Shift+Esc 
                        return 1;
                    if (kbh.vkCode == (int)Keys.Space && (int)Control.ModifierKeys == (int)Keys.Alt) //截获alt+空格 
                        return 1;
                    if (kbh.vkCode == 241)                  //截获F1 
                        return 1; if (kbh.vkCode == (int)Keys.Control && kbh.vkCode == (int)Keys.Alt && kbh.vkCode == (int)Keys.Delete)
                        return 1;
                    if ((int)Control.ModifierKeys == (int)Keys.Control + (int)Keys.Alt + (int)Keys.Delete)      //截获Ctrl+Alt+Delete 
                        return 1;
                    if ((int)Control.ModifierKeys == (int)Keys.Control + (int)Keys.Shift)      //截获Ctrl+Shift 
                        return 1;
                }
                return CallNextHookEx(hHook, nCode, wParam, lParam);
            }

            public void Dispose()
            {
                Close();
            }

        }

    钩子使用

    使用方法也很简单,在窗体初始化的时候,采用无边框并最大化窗体,然后启动钩子。

     在窗体关闭事件里关闭钩子,钩子使用一定要及时关闭。

    这样运行程序后,如果没有提供关闭程序的入门,似乎就只能重启系统了,大家测试时要注意保存电脑现有程序及文件。

    另外新一期的训练营很快就要开始了,先提前透漏一下内容

     

     

    -END-

     

  • 相关阅读:
    类型参数化
    scala 集合类型
    scala 列表List
    统计HDFS 上字节数据统计
    用scala 实现top N 排名
    scala 基础笔记
    Java 设计模式之 装饰者模式
    通过java api 读取sql 中数据(查询)
    leetcode 34. Search for a Range
    canvas画简单电路图
  • 原文地址:https://www.cnblogs.com/xiketang/p/15796980.html
Copyright © 2011-2022 走看看