Jni開發時,dll文件放置的路徑處理方式

剛到新公司,就接到一個棘手的任務。想了很多辦法,最後想使用Java COM橋來解決。JACOB是一個較成熟的開源項目,可以很方便的調用COM組件。搞過JNI的都知道,本地庫要放到系統path中,這樣,Java進程 在運行中才能找到本地庫並動態加載。我們可以通過環境變量System.getProperty("java.library.path")來查看當前 JVM搜索本地庫的路徑。

這時,就會遇到一個問題,部署應用的時候要記住將本地庫拷貝到環境變量path指定的路徑中。一般在 windows平臺上直接copy到C:/WINDOWS/System32目錄下了事。但要換一臺機器部署怎麼辦?除了要把Java程序拿過去,還要記 的把本地庫也copy到正確的目錄,真麻煩。於是想看看有什麼好辦法來解決這個問題。

首先,最容易想到的是,把本地庫和class文件放在一起,利用Class.getResource(str)找到路徑,然後加到環境java.library.path中:

代碼
  1. URL url = Foo.class.getResource("Foo.class");   
  2. String path = (new File(url.getPath())).getParent();   
  3. System.setProperty("java.library.path", path);  
<script type="text/javascript">render_code();</script>
看上去很好,但卻不能工作。查了一下ClassLoader的源代碼,原來它把搜索路徑定義爲靜態變量並只初始化一次,後面再設置java.library.path就沒有用了。ClassLoader代碼片斷:
代碼
  1. // The paths searched for libraries   
  2. static private String usr_paths[];   
  3. static private String sys_paths[];   
  4. ...   
  5. if (sys_paths == null) {   
  6.     usr_paths = initializePath("java.library.path");   
  7.     sys_paths = initializePath("sun.boot.library.path");   
  8. }  
<script type="text/javascript">render_code();</script>
正在一籌莫展是,翻看JACOB的源代碼,忽然有了驚喜的發現。
代碼
  1. try  
  2. {   
  3.     //Finds a stream to the dll. Change path/class if necessary   
  4.     InputStream inputStream = getClass().getResource("/jacob.dll").openStream();   
  5.     //Change name if necessary   
  6.     File temporaryDll = File.createTempFile("jacob"".dll");   
  7.     FileOutputStream outputStream = new FileOutputStream(temporaryDll);   
  8.     byte[] array = new byte[8192];   
  9.     for (int i = inputStream.read(array); i != -1; i = inputStream.read(array)) {   
  10.                 outputStream.write(array, 0, i);   
  11.         }   
  12.     outputStream.close();   
  13.     temporaryDll.deleteOnExit();   
  14.     System.load(temporaryDll.getPath());   
  15.     return true;   
  16. }   
  17. catch(Throwable e)   
  18. {   
  19.     e.printStackTrace();   
  20.     return false;   
  21. }  
<script type="text/javascript">render_code();</script>
高, 真是好辦法。和我一樣,把dll放在classpath中,用Class.getResource(str).openStream()讀取這個dll, 然後寫到temp目錄中,最後用System.load(path)來動態加載。多說一句,爲什麼得到了jacob.dll的URL不直接去加載呢?想想 看,如果把dll和class一起打成Jar包,ClassLoader還是不能加載本地庫,因爲System.load(path)需要的是dll的完 整路徑,但並不支持jar協議。還不明白,看看下面的代碼:
代碼
  1. URL url = Foo.class.getResource("/java/lang/String.class");   
  2. System.out.println(url.toExternalForm());   
  3. System.out.println(url.getFile());  
<script type="text/javascript">render_code();</script>
在我的機器上的運行結果是:
代碼
  1. jar:file:/D:/jdk1.5.0_06/jre/lib/rt.jar!/java/lang/String.class  
  2. file:/D:/jdk1.5.0_06/jre/lib/rt.jar!/java/lang/String.class  
<script type="text/javascript">render_code();</script>
ClassLoader中用new File(name),當然會找不到文件。同時,看看我的第一種方法,就算能設置成功環境java.library.path,如果dll是在jar包中,還是加載不了。

 

到現在,你是不是覺得問題已經解決了?還沒呢!jacob的很多源文件中已經寫了下面的代碼:

代碼
  1. static {   
  2.     System.loadLibrary("jacob");   
  3. }  
<script type="text/javascript">render_code();</script>
除非我去掉這一句重新編譯jacob的源代碼,否則系統還是會報錯。不過,既然有了上面的想法,稍微變通一下,就可以巧妙的解決。首先找到環境java.library.path,然後把dll拷貝到其中一個路徑中就行了。
代碼
  1. static {   
  2.     try {   
  3.         String libpath = System.getProperty("java.library.path");   
  4.         if ( libpath==null || libpath.length() == 0 ) {   
  5.             throw new RuntimeException("java.library.path is null");   
  6.         }   
  7.                
  8.         String path = null;   
  9.         StringTokenizer st = new StringTokenizer(libpath, System.getProperty("path.separator"));   
  10.         if ( st.hasMoreElements() ) {   
  11.             path = st.nextToken();   
  12.         } else {   
  13.             throw new RuntimeException("can not split library path:" + libpath);   
  14.         }   
  15.                
  16.         InputStream inputStream = Foo.class.getResource("jacob.dll").openStream();   
  17.         final File dllFile = new File(new File(path), "jacob.dll");   
  18.         if (!dllFile.exists()) {   
  19.             FileOutputStream outputStream = new FileOutputStream(dllFile);   
  20.             byte[] array = new byte[8192];   
  21.             for (int i = inputStream.read(array); i != -1; i = inputStream.read(array)) {   
  22.                 outputStream.write(array, 0, i);   
  23.             }   
  24.             outputStream.close();   
  25.         }   
  26.         //dllFile.deleteOnExit();      
  27.         Runtime.getRuntime().addShutdownHook(new Thread(){   
  28.             public void run() {   
  29.                 if ( dllFile.exists() ) {   
  30.                     boolean delete = dllFile.delete();   
  31.                     System.out.println("delete : " + delete);   
  32.                 }   
  33.             }   
  34.         });   
  35.         } catch (Throwable e) {   
  36.             throw new RuntimeException("load jacob.dll error!", e);   
  37.     }   
  38. }  
<script type="text/javascript">render_code();</script>
唯一的美中不足,在系統關閉的時候刪除dll總是不能成功,試了兩種辦法都不行。想想也對,dll正被程序使用,當然不能刪除。翻了一下API,Java好像沒用提供unload本地庫的功能,只好做罷。
解決了這麼個小問題,羅羅嗦嗦一大篇,罪過罪過。後來這個項目又沒有使用jacob,真對不起各位觀衆。 
 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章