剛到新公司,就接到一個棘手的任務。想了很多辦法,最後想使用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中:
- URL url = Foo.class.getResource("Foo.class");
- String path = (new File(url.getPath())).getParent();
- System.setProperty("java.library.path", path);
看上去很好,但卻不能工作。查了一下ClassLoader的源代碼,原來它把搜索路徑定義爲靜態變量並只初始化一次,後面再設置java.library.path就沒有用了。ClassLoader代碼片斷:
- // The paths searched for libraries
- static private String usr_paths[];
- static private String sys_paths[];
- ...
- if (sys_paths == null) {
- usr_paths = initializePath("java.library.path");
- sys_paths = initializePath("sun.boot.library.path");
- }
正在一籌莫展是,翻看JACOB的源代碼,忽然有了驚喜的發現。
- try
- {
- //Finds a stream to the dll. Change path/class if necessary
- InputStream inputStream = getClass().getResource("/jacob.dll").openStream();
- //Change name if necessary
- File temporaryDll = File.createTempFile("jacob", ".dll");
- FileOutputStream outputStream = new FileOutputStream(temporaryDll);
- byte[] array = new byte[8192];
- for (int i = inputStream.read(array); i != -1; i = inputStream.read(array)) {
- outputStream.write(array, 0, i);
- }
- outputStream.close();
- temporaryDll.deleteOnExit();
- System.load(temporaryDll.getPath());
- return true;
- }
- catch(Throwable e)
- {
- e.printStackTrace();
- return false;
- }
高, 真是好辦法。和我一樣,把dll放在classpath中,用Class.getResource(str).openStream()讀取這個dll, 然後寫到temp目錄中,最後用System.load(path)來動態加載。多說一句,爲什麼得到了jacob.dll的URL不直接去加載呢?想想 看,如果把dll和class一起打成Jar包,ClassLoader還是不能加載本地庫,因爲System.load(path)需要的是dll的完 整路徑,但並不支持jar協議。還不明白,看看下面的代碼:
- URL url = Foo.class.getResource("/java/lang/String.class");
- System.out.println(url.toExternalForm());
- System.out.println(url.getFile());
在我的機器上的運行結果是:
- jar:file:/D:/jdk1.5.0_06/jre/lib/rt.jar!/java/lang/String.class
- file:/D:/jdk1.5.0_06/jre/lib/rt.jar!/java/lang/String.class
ClassLoader中用new File(name),當然會找不到文件。同時,看看我的第一種方法,就算能設置成功環境java.library.path,如果dll是在jar包中,還是加載不了。
到現在,你是不是覺得問題已經解決了?還沒呢!jacob的很多源文件中已經寫了下面的代碼:
- static {
- System.loadLibrary("jacob");
- }
除非我去掉這一句重新編譯jacob的源代碼,否則系統還是會報錯。不過,既然有了上面的想法,稍微變通一下,就可以巧妙的解決。首先找到環境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正被程序使用,當然不能刪除。翻了一下API,Java好像沒用提供unload本地庫的功能,只好做罷。
解決了這麼個小問題,羅羅嗦嗦一大篇,罪過罪過。後來這個項目又沒有使用jacob,真對不起各位觀衆。