1. 程式人生 > >如何解決java.library.path屬性在程式碼中設定不生效問題

如何解決java.library.path屬性在程式碼中設定不生效問題

前兩天想的螢幕抓圖的程式,想給加個系統全域性的熱鍵,也就是說,在程式不處於當前啟用下也可響應鍵盤事件。通過java自己應該是實現不了了。
想到使用JNI,還好找到一篇介紹使用JIntellitype的文章,寫得挺好的,下面的是連結:
http://walsece.iteye.com/blog/191169

可是在使用上面文章介紹的動態庫時,遇上了載入問題,也就是將動態庫放到什麼位置合適?文章中說可以放到System32目錄,但我不想這樣做,我更希望的是可以自己定製dll庫的位置~~

沒有將lib放到System32目錄,而是放到了自己的lib目錄,程式在啟動的過程中會報下面的錯誤:
java.lang.UnsatisfiedLinkError: no JIntellitype in java.library.path

根據上面的提示資訊在VM引數處通過-Djava.library.path將載入路徑指定到自己的lib目錄後,程式可以正常啟動。

這種方式不是太好,因為要手動的去指定虛擬機器引數,於是想通過System類的setProperty函式來在程式碼中動態的改變一下java.library.path的值。
使用

Java程式碼 複製程式碼 收藏程式碼
  1. System.setProperty("java.library.path""./lib");     
System.setProperty("java.library.path", "./lib");   

後,啟動程式總是報錯"no JIntellitype in java.library.path"
了半天沒有想明白,為什麼程式碼的設定就不起作用,而在虛擬機器引數處指定就是好的?

Google了半天才知道原因,這裡整理一下,以方便以後查詢。
程式碼中設定不起作用,主要是因為java.library.path只在jvm啟動時讀取一次,其他情況下的修改不會起作用的。可以參考下面的這個bug:

http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4280189

原因和ClassLoader的實現有關係,
ClassLoader.loadLibrary() method:

Java程式碼 複製程式碼 收藏程式碼
  1. if (sys_paths == null) {      
  2.         usr_paths = initializePath("java.library.path");      
  3.         sys_paths = initializePath("sun.boot.library.path");      
  4. }     
if (sys_paths == null) {   
        usr_paths = initializePath("java.library.path");   
        sys_paths = initializePath("sun.boot.library.path");   
}   

系統快取了java.library.path的值,並且一直都會是第一次載入時的值。有人提到了下面的修改方法,

Java程式碼 複製程式碼 收藏程式碼
  1. if (sys_paths == null) {      
  2.        sys_paths = initializePath("sun.boot.library.path");      
  3. }      
  4. usr_paths = initializePath("java.library.path");     
if (sys_paths == null) {   
       sys_paths = initializePath("sun.boot.library.path");   
}   
usr_paths = initializePath("java.library.path");   

但是從2002年到現在Sun一直都沒有改,不知道出於什麼原因考慮的。

有問題,就會有人解決問題,antony_miguel在一篇文章中,使用java的反射機制,完成了對於ClassLoader類中的usr_paths變數的動態修改,

Java程式碼 複製程式碼 收藏程式碼
  1. publicstaticvoid addDir(String s) throws IOException {      
  2. try {      
  3.         Field field = ClassLoader.class.getDeclaredField("usr_paths");      
  4.         field.setAccessible(true);      
  5.         String[] paths = (String[])field.get(null);      
  6. for (int i = 0; i < paths.length; i++) {      
  7. if (s.equals(paths[i])) {      
  8. return;      
  9.             }      
  10.         }      
  11.         String[] tmp = new String[paths.length+1];      
  12.         System.arraycopy(paths,0,tmp,0,paths.length);      
  13.         tmp[paths.length] = s;      
  14.         field.set(null,tmp);      
  15.     } catch (IllegalAccessException e) {      
  16. thrownew IOException("Failed to get permissions to set library path");      
  17.     } catch (NoSuchFieldException e) {      
  18. thrownew IOException("Failed to get field handle to set library path");      
  19.     }      
  20. }     
public static void addDir(String s) throws IOException {   
    try {   
        Field field = ClassLoader.class.getDeclaredField("usr_paths");   
        field.setAccessible(true);   
        String[] paths = (String[])field.get(null);   
        for (int i = 0; i < paths.length; i++) {   
            if (s.equals(paths[i])) {   
                return;   
            }   
        }   
        String[] tmp = new String[paths.length+1];   
        System.arraycopy(paths,0,tmp,0,paths.length);   
        tmp[paths.length] = s;   
        field.set(null,tmp);   
    } catch (IllegalAccessException e) {   
        throw new IOException("Failed to get permissions to set library path");   
    } catch (NoSuchFieldException e) {   
        throw new IOException("Failed to get field handle to set library path");   
    }   
}   

文章也同時指出了這種實現的侷限性,和jvm的實現強關聯,只要jvm實現不是用的變數usr_paths來儲存java.library.path的值,這個方法就不能用了。
但是隻要知道原始碼,小小的改動就應該可以實現了。