如何在你的java程序中注册系统级热键

原创 2011年11月01日 10:14:44

 所谓系统级热键就是指一组快捷键,不论当前系统焦点在哪个程序中,只要按下该键,程序就能够捕捉该事件并进行相关处理。该功能在应用程序中是非常有用的,比如系统自带的 
“win+L”自动锁屏,QQ中默认的“ctrl+alt+Z”自动打开当前的消息窗口等等。 

Java中的事件监听机制虽然功能强大,但是当系统焦点脱离该程序时也无能为力。要实现该功能必须调用系统的钩子函数,因此在java中也必须通过jni调用来实现,但是对于不熟悉系统函数或者其他编成语言的朋友来说却是个难题。 

以前实现类似的功能都是自己通过c调用系统的钩子函数然后再通过jni调用,自己写的东西只要能满足简单的需求即可,因此功能和程序结构也比较简单。后来在国外的一个网站上发现了一个组件“jintellitype”帮我们封装了绝大部分的功能,而且结构上也采用java中的事件监听机制,我们只要在程序中通过注册即可实现,下面是一个简单的例子:

 

import com.melloware.jintellitype.HotkeyListener;   
import com.melloware.jintellitype.JIntellitype;   
   
public class HotKey implements HotkeyListener {   
    static final int KEY_1 = 88;   
    static final int KEY_2 = 89;   
    static final int KEY_3 = 90;   
   
    /**  
     * 该方法负责监听注册的系统热键事件  
     *  
     * @param key:触发的热键标识  
     */   
    public void onHotKey(int key) {   
        switch (key) {   
            case KEY_1:   
                System.out.println("ctrl+alt+I 按下.........");   
                break;   
            case KEY_2:   
                System.out.println("ctrl+alt+O 按下.........");   
                break;   
            case KEY_3:   
                System.out.println("系统退出..........");   
                destroy();   
        }   
   
    }   
   
   
    /**  
     * 解除注册并退出  
     */   
    void destroy() {   
        JIntellitype.getInstance().unregisterHotKey(KEY_1);   
        JIntellitype.getInstance().unregisterHotKey(KEY_2);   
        JIntellitype.getInstance().unregisterHotKey(KEY_3);   
        System.exit(0);   
    }   
   
    /**  
     * 初始化热键并注册监听事件  
     */   
    void initHotkey() {   
        //参数KEY_1表示改组热键组合的标识,第二个参数表示组合键,如果没有则为0,该热键对应ctrl+alt+I   
        JIntellitype.getInstance().registerHotKey(KEY_1, JIntellitype.MOD_CONTROL + JIntellitype.MOD_ALT,   
                (int) 'I');   
        JIntellitype.getInstance().registerHotKey(KEY_2, JIntellitype.MOD_CONTROL + JIntellitype.MOD_ALT,   
                (int) 'O');   
        JIntellitype.getInstance().registerHotKey(KEY_3, JIntellitype.MOD_CONTROL + JIntellitype.MOD_ALT,   
                (int) 'X');   
   
        JIntellitype.getInstance().addHotKeyListener(this);   
    }   
   
    public static void main(String[] args) {   
        HotKey key = new HotKey();   
        key.initHotkey();   
   
        //下面模拟长时间执行的任务   
        while (true) {   
            try {   
                Thread.sleep(10000);   
            } catch (Exception ex) {   
                break;   
            }   
        }   
    }   
}   

也可参考:http://www.cppblog.com/biao/archive/2010/01/31/106859.html

注意:
1. 注册热键时必须是大写的。像 (int'I'
2. 

因为用到了JIntellitype,所以遇到如何加载dll的问题。
首先,我的目标是程序只有一个exe文件。研究了一翻exe4j,没发现把dll打进exe里的方法。
退而求次,准备把dll打包到jar中,但当dll打包进jar后,系统无法加载。
解决思路:
1. 获取jar中dll文件内容:class.getResource()
2. 把内容拷贝出来
3. 加载拷贝出来的dll文件


把dll拷贝到哪里,这也是个问题,
参考    http://hcx-2008.iteye.com/blog/285632 
先获取 java.library.path ,再把 dll拷贝到此路径下。这种方法不错。
static {   
    try {   
        String libpath = System.getProperty("java.library.path");   
        if ( libpath==null || libpath.length() == 0 ) {   
            throw new RuntimeException("java.library.path is null");   
        }   
               
        String path = null;   
        StringTokenizer st = new StringTokenizer(libpath, System.getProperty("path.separator"));   
        if ( st.hasMoreElements() ) {   
            path = st.nextToken();   
        } else {   
            throw new RuntimeException("can not split library path:" + libpath);   
        }   
               
        InputStream inputStream = Foo.class.getResource("jacob.dll").openStream();   
        final File dllFile = new File(new File(path), "jacob.dll");   
        if (!dllFile.exists()) {   
            FileOutputStream outputStream = new FileOutputStream(dllFile);   
            byte[] array = new byte[8192];   
            for (int i = inputStream.read(array); i != -1; i = inputStream.read(array)) {   
                outputStream.write(array, 0, i);   
            }   
            outputStream.close();   
        }   
        //dllFile.deleteOnExit();      
        Runtime.getRuntime().addShutdownHook(new Thread(){   
            public void run() {   
                if ( dllFile.exists() ) {   
                    boolean delete = dllFile.delete();   
                    System.out.println("delete : " + delete);   
                }   
            }   
        });   
        } catch (Throwable e) {   
            throw new RuntimeException("load jacob.dll error!", e);   
    }   
}  


 

关于不能删除dll文件问题,要先卸载它:
//卸载dll
                try {
                    ClassLoader classLoader = this.getClass().getClassLoader();
                    Field field = ClassLoader.class.getDeclaredField("nativeLibraries");
                    field.setAccessible(true);
                    Vector libs = (Vector) field.get(classLoader);
                    Iterator it = libs.iterator();
                    Object o;
                    while (it.hasNext()) {
                        o = it.next();
                        Method finalize = o.getClass().getDeclaredMethod("finalize", new Class[0]);
                        finalize.setAccessible(true);
                        finalize.invoke(o, new Object[0]);
                    }
                } catch (Exception ex) {
                    log.error("卸载dll文件出错,需要重启服务器!", ex);
                    throw new RuntimeException(ex);
                }
原文: http://wiflish.iteye.com/blog/400370 


经测试,此段代码有问题,没找出问题原因,以后再研究了。。。



 

章鱼哥出品—VB.NET 自定义快捷键使用详解之全局热键

如何设置VB.NET 窗体的全局热键(快捷键),
  • zhangyubishoulin
  • zhangyubishoulin
  • 2014年10月06日 01:49
  • 4205

MFC实现全局热键

要想使用全局热键的话,需要用到两个函数: BOOL WINAPI RegisterHotKey( __in_opt HWND hWnd, __in int id, __...
  • ccfxue
  • ccfxue
  • 2016年05月10日 16:53
  • 1117

Windows编程实现自定义全局热键

一、实现全局热键主要使用以下API(详情查看MSDN):- 获取热键标识:GlobalAddAtom - 注册热键:RegisterHotKey - 注销热键:UnregisterHotKey - 释...
  • zhoufuxinzfx
  • zhoufuxinzfx
  • 2016年03月09日 16:22
  • 2840

如何实现全局热键

要实现系统范围内的热键
  • wwkaven
  • wwkaven
  • 2014年06月14日 12:39
  • 2288

WPF的热键实现

项目开发中,需要实现对于热键的支持,实际上有两种方案来实现,一种是采用WPF的方式,即命令绑定,利用InputBinding实现ICommand的快捷键绑定处理,另一种是调用系统API注册热键的方式....
  • u013376417
  • u013376417
  • 2015年03月13日 13:56
  • 1593

C#全局热键设置与窗体热键设置实例

1、窗体热键 首先要设置主窗体KeyPreview为true,可直接在属性中进行设置, 或者在窗体加载中设置: this.KeyPreview = true; 然后添加窗体KeyDown事件,如...
  • cuoban
  • cuoban
  • 2016年02月26日 16:16
  • 3392

Win7 禁止Ctrl+Alt+Del、Win+L等任意系统热键

标 题: 【原创】Win7 修改Winlogon.exe进程一个字节禁止Ctrl+Alt+Del、Win+L等任意系统热键 作 者: heiheiabcd 时 间: 2012-12-01,10...
  • linfei2707
  • linfei2707
  • 2014年05月07日 18:13
  • 7888

[JNI]用JAVA实现全局快捷键

基本思路:使用WIN API实现一个底层键盘钩子,监听按键事件。如果需要的快捷键被触发,则弹出相应的窗口。 找到了http://www.jotschi.de/?p=90 这个代码基本上实现了我的要...
  • ycb1689
  • ycb1689
  • 2013年03月05日 11:44
  • 1942

C#实现全局快捷键(系统热键)响应(转)

在应用中,我们可能会需要实现像Ctrl+C复制、Ctrl+V粘贴这样的快捷键,本文简单介绍了它的实现,并给出了一个实现类。 (1)建立一个类文件,命名为HotKey.cs,代码如下:...
  • zxj19951029
  • zxj19951029
  • 2014年10月18日 23:43
  • 459

C#屏蔽系统热键Ctrl+Alt+Delete的代码尝试。

最近在做一款小软件,需要锁定用户的输入,包括系统热键(Ctrl+Alt+Delete),在网络中寻找良久,发现这些不错的代码。 经过仔细整理,现将这些代码公布,有兴趣的拿去用用。 搜集的过程有些艰...
  • llw01
  • llw01
  • 2013年07月15日 15:45
  • 1902
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:如何在你的java程序中注册系统级热键
举报原因:
原因补充:

(最多只允许输入30个字)