看板 java 關於我們 聯絡資訊
今天在寫程式的時候 我希望讓我程式有類似plug-in的功能 所以我試著用URLClassLoader去實作這個功能 首先我測試了一下 先用eclipse開了一個project A 裡面一個interface interface CLI2 { public String say(); } 而存在一個class 去實作他 public class CLTest implements CLI2 { static{ System.out.println("Now!Class Load!"); } @Override public String say() { System.out.println("Hello I am a happy boy!"); return "Hello"; } } 我把他輸出成tester.jar 之後我再開了一個project B 裡面是引用他的程式碼 URL url1 = new URL("file:c:/TC/"); URLClassLoader urlClassLoader1 = new URLClassLoader(new URL[] {url1} ,Thread.currentThread().getContextClassLoader()); Class c2 = urlClassLoader1.loadClass("CLTest"); Object o=c2.newInstance(); CLI2 ci=(CLI2)o; ci.say(); ----------------------------- Project B裡面有隻相同的inteface CLI2,他是希望用來轉型 跑下去後出現了error Exception in thread "main" java.lang.IllegalAccessError: class CLTest cannot access its superinterface CLI2 上網查了一下原因,似乎是因為當前的classLoader沒辦法存取他的interface 我把程式作了修改,URLClassLoader第二個參數改成null URL url1 = new URL("file:c:/TC/"); URLClassLoader urlClassLoader1 = new URLClassLoader(new URL[] {url1} ,null); Class c2 = urlClassLoader1.loadClass("CLTest"); Object o=c2.newInstance(); CLI2 ci=(CLI2)o; ci.say(); 雖然解決了第一個問題卻延伸出另一個問題 在CLI2 ci=(CLI2)o;的地方出現轉型錯誤的exception Exception in thread "main" java.lang.ClassCastException: CLTest cannot be cast to CLI2 後來上網查了一下資料 才知道他型別判定是除了class name跟package name 還包括ClassLoader,我的CLI2是AppClassLoader載入的 o則是我創造的URLClassLoader載入的 我的問題是,我要如何在Project B保有inteface CLI2的情況下 可以讓他跟我用URLClassLoade載入的CLTest物件相容 或是用Thread.currentThread().getContextClassLoader() 讓他可以抓到CLT2的interface部出現error 請高手指教 -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 122.126.48.73 ※ 編輯: del680202 來自: 122.126.48.73 (12/14 15:03)
PsMonkey:不太懂.. 為甚麼需要兩個 CLI2? 12/14 15:24
一個是我Project A實作用到到 一個是Project B產生實體拿來轉型用 因為是不同的Project,或許有更好的方法吧 ※ 編輯: del680202 來自: 122.126.48.73 (12/14 15:30)
snowlike:PA和PB實作的CLI2要同個,這也是為什麼需import避免撞名 12/14 15:55
PA跟PB有的CLI2是同樣的內容 我有試著PA只輸出CLTest的.class檔 但是會出現NoClassDefFoundError的錯誤 沒有CLI2的定義 因為我希望是說,我主程式放一個地方 plug-in可以放另一個地方 以後只要更新plug-in就能更新程式,而我plug-in只需要實作相同介面就可以了 ※ 編輯: del680202 來自: 122.126.48.73 (12/14 16:07) ////////////////////////////////////////////////////////////////////// 問題解決了,snowlike大的話給了我一點啟示 我Project A原先兩隻程式都是放在default package 我把interface的部分放在另一個package org.cusano 而Project B同樣維護著一份interface,而我在同樣的創造一個package org.cusano在PB 之後PA的實作部分跟PB的引用部分都去import org.cusano.*; 這樣就解決了衝突的問題 感激不盡 ※ 編輯: del680202 來自: 122.126.48.73 (12/14 16:24) ※ 編輯: del680202 來自: 122.126.48.73 (12/14 16:25)
sbrhsieh:在兩個地方維護必須相同的兩份碼是給自己找麻煩... 12/14 16:26
del680202:只維護相同的interface 主要是想針對interface就好 12/14 16:27
del680202:不過或許有更好的方法吧 暫時沒想到 12/14 16:30