跳到主要內容

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

留言

這個網誌中的熱門文章

iframe DOM 被移動造成重新載入

如果用 javascript 去搬動 iframe DOM 的位置, 瀏覽器會將其內容重新載入,這是現有 HTML 規格 > When an iframe element is inserted into a document that has a browsing context, the user agent must create a nested browsing context, and then process the iframe attributes for the "first time". 範例: http://jsfiddle.net/pZ23B/ 測試結果: * Safari 3.1 / Win: reload * Opera 9.5 / Win: reload * IE10: reload * IE7 / IE8: not reload (部份摘自 https://bugzilla.mozilla.org/show_bug.cgi?id=254144 ) 參考: * https://bugzilla.mozilla.org/show_bug.cgi?id=254144

JavaScript 關掉瀏覽器頁面

如果你直接呼叫 window.close() 來關掉目前的頁面的話,你應該會在 console 看到以下訊息: Scripts may close only the windows that were opened by it. ( 我在 Chrome, Firefox, IE 11 都試過) 主因就是你並沒有用 JavaScript 開啟這個頁面,所以也不能用 JavaScript 關掉它。這是 HTML window.close() 的規格 規定,各家瀏覽器應該都遵循。 隨著瀏覽器安全性增加,以下方法已經不適用了,可參考 這個討論 。 有個變通的辦法就是: window.open(location, '_self').close();

Web Dynpro 前後端資料流動機制 (Dataflow)

在Web Dynpro中提供3種資料流機制[1],只要適當地設定,可以不用寫程式就將畫面、中間層控制器(controller)到後端模型物件的資料自動化地、牢靠地綁在一起,使得不管前後端某一方有資料變動,變動部份都會自動地流動來保持一致性,使得前後端都能存取到同一份資料。 context 關鍵元件是 context,每個controller都有一份屬於自己的context,它扮演MVC架構中的M (model),web dynpro的實做方式比較像是該controller的「資料空間」,它由node(資料節點)與attribute(資料屬性)組成,controller可以透過wdContext這個預先產生好的 shortcut variable(捷徑變數)去取得context的資料內容。 context中必須要建立node才可以儲存資料,一個node代表一個collection(集合物件)裡面仍可以含有node, attribute,node裡面的一份資料實體就是一個element,一個node可以有一個或多個element(這部份可以透過cardinality property設定),其結構就是該node所包含的結構。 data binding 此種機制可以將UI元件的資料跟context中的某個node或attribute綁定在一起,context中的改變會自動 更新到UI元件上,UI元件的改變也會自動寫入到綁定的context node(or attribute)中。通常UI元件所綁定的node(attribute)是由component controller對應過來的。 context mapping 每一個controller都有屬於自己的context(資料空間)如果要達到彼此共享資料,則要透過context mapping機制,一旦mapping設定好,則會在另一個context產生一個同樣的結構的node,兩邊的controller會存取到的是同一分資料,任何一邊的資料更動都會散佈到設定好mapping的node。 不過在web dynpro裡,只允許將custom controller or component controller的node對應到view controller去,不允許從view controller對應回來,主因是嚴格遵守MVC...