跳到主要內容

JNI(Java Native Interface) Learning Notes

JNI Learning Notes

要從Java中呼叫第三方C的DLL, cstat.dll,該程式沒有source code無法重新編譯。
因JNI的方式來呼叫的話,C的函數名稱有固定的規則,因為無cstat.dll的原始檔可以修改,因此要用C/C++寫一個中間程式來呼叫cstat.dll。
架構為
Java Class -> intermedia c program -> cstat.dll
( ->代表函數呼叫 )

產生C header file
javah -jni Hello
produce Hello.h

C compiler
Download free cygwin
安裝時要選擇gcc的套件一併安裝,才有gcc可用。

編譯出object file,
要求編譯系統不要使用 cygwin的DLL
gcc -c -I C:/j2sdk1.4.2_13/include -I C:/j2sdk1.4.2_13/include/win32 -mno-cygwin Sample1.c
-c 執行編譯
-I 指定jni.h的目錄位置與作業系統相關的header file位置

Build DLL from .o file
gcc -shared -o
Sample1.dll Sample1.o
製作 DLL 時要求編譯系統不要使用 cygwin 的 DLL

gcc -mno-cygwin -shared -o HelloImp.dll HelloImp.o

Build DLL directly from .c file
gcc -shared -mno-cygwin -I C:/j2sdk1.4.2_13/include -I C:/j2sdk1.4.2_13/include/win32 Arguments.c -o Arguments.dll

Final correct command with options
gcc -shared -mno-cygwin -Wl,--kill-at -I C:/j2sdk1.4.2_13/include -I C:/j2sdk1.4.2_13/include/win32 Arguments.c -o Arguments.dll

-Wl,option
把option傳遞給連接器,如果option中含有逗號,就在逗號處分割成多個選項

from JavaWorld 有人寫到:
在Windows下用gcc編譯可被JNI呼叫的dll檔要這樣下:
gcc -Wall -D_JNI_IMPLEMENTATION_ -Wl,--kill-at -I/java/include -I/java/include/win32 -shared -o JavaImp.dll someJavaImp.c
我用的是MinGW

經過我測試 不加
-D_JNI_IMPLEMENTATION_也可以運作,事實上我在gcc的手冊中找不到這個option
我也找不到--kill-at這個option。

Load DLL in C++

#include
HINSTANCE fhandle= LoadLibrary("CSTAT.DLL");

HINSTANCE is defined in windows.h
[BCB help]
HINSTANCE, Indicates the handle provided by Windows for an application or library.
The LoadLibrary function maps the specified executable module into the address space of the calling process.
If the function succeeds, the return value is a handle to the module.
If the function fails, the return value is NULL.
typedef __stdcall(*Functiontype)(int) //first define function type, int is received argutment list
afunction = (
Functiontype)GetProcAddress(fhandle, "afunction_name");[BCB help]
The GetProcAddress function returns the address of the specified exported dynamic-link library (DLL) function.
FARPROC GetProcAddress(
HMODULE hModule, // handle to DLL module
LPCSTR lpProcName // name of function
);

問題
找不到程序進入點,在cygwin1.dll
此問題可透過編譯時加上-mno-cygwin來移除cygwin DLL的相依性


Exception in thread "main" java.lang.UnsatisfiedLinkError: hello
at Hello.main(Hello.java:9)


Exception in thread "main" java.lang.UnsatisfiedLinkError: no Sample1 in java.li
brary.path
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1290)
at java.lang.Runtime.loadLibrary0(Runtime.java:473)
at java.lang.System.loadLibrary(System.java:777)
at Sample1.main(Sample1.java:10)
找不到所需的dll,確認系統環境變數PATH中有包含dll所在的目錄。

Exception in thread "main" java.lang.UnsatisfiedLinkError: D:\Hawk\workspace\Jni
\Sample1.dll: The specified procedure could not be found
at java.lang.ClassLoader$NativeLibrary.load(Native Method)
at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1360)
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1284)
at java.lang.Runtime.loadLibrary0(Runtime.java:473)
at java.lang.System.loadLibrary(System.java:777)
at Sample1.main(Sample1.java:10)

相關書籍:
Java深度歷險, 王森, McGrawHill

留言

這個網誌中的熱門文章

Code Complete, 2ed (中譯:軟體建構之道) 讀後心得

Code Complete, 2ed (中譯:軟體建構之道) 讀後心得 雖然以前在學校有修過軟體工程,但是說實在現在我對建構軟體的整個概念都來自於兩本書:「人月神話」(The Mythical Man-Month)、Code Complete[1]。我畢業後第一份工作是軟體研發工程師,我們公司所開發的系統大約有近百位工程師共同維護,當然各有依功能與專業知識將系統劃分不同的區 塊來維護,在面對大型的系統與這麼多人寫程式是以前在學校沒有遇過的,因此也開始思考怎樣才是好的軟體開發方式。從大學好友那探聽到了Code Complete這本書,花了上千元買了英文版,(現在中文版已出,簡體版聽說只要三分之一的價錢)認真讀了幾篇之後,真是如獲至寶,於是自己讀過後,還跟同事組成讀書會一同研討這本書。 作者 Steve McConnell (個人Blog http://www.stevemcconnell.com/ )著作不算量大,但是我看過的幾本質量都很好,印象中他曾經在微軟工作過,後來成立一家軟體顧問公司 Construx Software 。 這本書(書本網站 http://cc2e.com/Default.aspx )的討論主題很廣泛,但是並不空泛,從最小的、最基本的如何寫出好的程式, 例如函式(routine)、註解 (comment)、變數的命名等,也談整個軟體建構的過程除錯、測試,最後也談到軟體工程師個人能力與技術生涯的發展,算是各方面都有提及。但是我比較 建議有半年以上的實際程式經驗來讀這本書比較適合,我自認為如果我在學生時代接觸本書,可能沒有什麼太大的感觸,因為面對的都是小格局、人數少的小系統, 書中所探討的原則與例子,很多時候沒有團隊開發或大型系統的經驗,很難體會。但稍有經驗的人,看完之後保證在程式撰寫與軟體開發方面的觀念會有很大的提 昇。 Table of Content Laying the Foundation 1 Welcome to Software Construction 2 Metaphors for a Richer Understanding of Software Development 3 Measure Twice, Cut Once: Upstream Prerequisites 4 Key...

innerHTML 的安全風險

問題 若你要實作一個使用者輸入的功能,並把輸入的內容顯示在網頁上時,應避免把使用者輸入的值直接指定給某個 element 的 innerHTML 。因為若是使用者輸入包含 HTML、JavaScript, innerHTML 就會解析並執行: var userInput = '<img src="javascript:alert("XSS")">' element . innerHTML = userInput ; 因此可能 user A 的惡意輸入,會被系統顯示到 user B 的畫面上,進而執行特定成程式碼而造成 XSS 攻擊 。 解法 一般會先想到 encode HTML,但是如果只是要顯示文字,將使用者輸入值指定給 textContent 會更簡單,無需 encode,該值會被當成純文字處理,並不會執行 javascript。 var userInput = '<img src="javascript:alert("XSS")">' element . textContent = userInput ;

ZK 教學 - 常見錯誤用法 06 - 慎選 data binding 語法

慎選 data binding 語法 最常用的語法有以下 4 種: @init: 只從 ViewModel 載入一次 @load:載入並追蹤 ViewModel 變化 @save:寫入 ViewModel1 @bind:(雙向)等於 @load + @save 後面 3 種因為需要持續維持(追蹤)元件與 ViewModel 的綁定關係,伺服器需要建立 tracking node,因此成本較高。 依對系統的負擔排名如下: 因此如果沒有動態變化的需要,請使用 @init 。 例如在 Listbox , Grid , Tree 內的 <template> 若只是顯示資料,只需用 @init ,可以免去大量 tracking node 所需的記憶體。 例如: < grid width = " 400px " model = " @init(vm.itemList) " > < columns > < column label = " index " /> < column label = " name " /> </ columns > < template name = " model " > < row > < label value = " @init(forEachStatus.index) " /> < label value = " @init(each.name) " /> </ row > </ template > </ grid >