要在java中調(diào)用c語(yǔ)言的庫(kù),需要使用Java提供了JNI。\x0d\x0a舉例說(shuō)明\x0d\x0a在c語(yǔ)言中定義一個(gè)voidsayHello()函數(shù)(打印HelloWorld);然后在Java中調(diào)用這個(gè)函數(shù)顯示HelloWord.\x0d\x0a現(xiàn)在分別從Java和C語(yǔ)言兩部分說(shuō)明:\x0d\x0a1.Java部分\x0d\x0a首先定義一個(gè)HelloNative,在其中申明sayHello函數(shù),函數(shù)要申明為Native類型的.如下:\x0d\x0apublicclassHelloNative{\x0d\x0apublicnativevoidsayHello();\x0d\x0a}\x0d\x0a\x0d\x0a編譯這個(gè)類,生成class文件:\x0d\x0ajavacHelloWorld.java\x0d\x0a\x0d\x0a利用javah生成需要的h文件\x0d\x0ajavahHelloNative\x0d\x0a\x0d\x0a生成的h文件大概如下:\x0d\x0a\x0d\x0a/*DONOTEDITTHISFILE-itismachinegenerated*/\x0d\x0a#include\x0d\x0a/*HeaderforclassHelloNative*/\x0d\x0a\x0d\x0a#ifndef_Included_HelloNative\x0d\x0a#define_Included_HelloNative\x0d\x0a#ifdef__cplusplus\x0d\x0aextern"C"{\x0d\x0a#endif\x0d\x0a/*\x0d\x0a*Class:HelloNative\x0d\x0a*Method:sayHello\x0d\x0a*Signature:()V\x0d\x0a*/\x0d\x0aJNIEXPORTvoidJNICALLJava_HelloNative_sayHello\x0d\x0a(JNIEnv*,jobject);\x0d\x0a\x0d\x0a#ifdef__cplusplus\x0d\x0a}\x0d\x0a#endif\x0d\x0a#endif\x0d\x0a\x0d\x0a可以看一下上面自動(dòng)生成的程序,程序include了jni.h,這個(gè)頭文件在$JAVA_HOME下的include文件夾下.還可以發(fā)現(xiàn)生成的函數(shù)名是在之前的函數(shù)名前面加上了Java_HelloNative。\x0d\x0a2.C語(yǔ)言部分\x0d\x0a根據(jù)上面生成的h文件編寫(xiě)相應(yīng)的代碼實(shí)現(xiàn),建立一個(gè)HelloNative.cpp用來(lái)實(shí)現(xiàn)顯示HelloWorld的函數(shù).如下:\x0d\x0a\x0d\x0a#include\x0d\x0a#include"HelloNative.h"\x0d\x0a\x0d\x0aJNIEXPORTvoidJNICALLJava_HelloNative_sayHello(JNIEnv*,jobject)\x0d\x0a{\x0d\x0aprintf("HelloWorld!\n");\x0d\x0a}\x0d\x0a\x0d\x0a代碼編寫(xiě)完成之后,我們?cè)儆胓cc編譯成庫(kù)文件,命令如下;\x0d\x0agcc-fPIC-I/usr/lib/jvm/java-7-openjdk-i386/include-I/usr/lib/jvm/java-7-openjdk-i386/include/linux-shared-olibHelloNative.soHelloNative.cpp\x0d\x0a\x0d\x0a這樣就會(huì)在當(dāng)前目錄下生成一個(gè)libHelloNative.so的庫(kù)文件.這時(shí)需要的庫(kù)已經(jīng)生成,在C語(yǔ)言下的工作已經(jīng)完成了.\x0d\x0a接下來(lái)需要在Java中編寫(xiě)一個(gè)程序測(cè)試一下.在程序前,需要將我們的庫(kù)載入進(jìn)去.載入的方法是調(diào)用Java的System.loadLibrary("HelloNative");\x0d\x0a\x0d\x0apublicclassTestNative\x0d\x0a{\x0d\x0astatic{\x0d\x0atry{\x0d\x0aSystem.loadLibrary("HelloNative");\x0d\x0a}\x0d\x0acatch(UnsatisfiedLinkErrore){\x0d\x0aSystem.out.println("Cannotloadhellolibrary:\n"+e.toString());\x0d\x0a}\x0d\x0a}\x0d\x0apublicstaticvoidmain(String[]args){\x0d\x0aHelloNativetest=newHelloNative();\x0d\x0atest.sayHello();\x0d\x0a}\x0d\x0a}\x0d\x0a\x0d\x0a但是再編譯后,運(yùn)行的時(shí)候,問(wèn)題又出現(xiàn)了.\x0d\x0aCannotloadhellolibrary:\x0d\x0ajava.lang.UnsatisfiedLinkError:noHelloNativeinjava.library.path\x0d\x0aExceptioninthread"main"java.lang.UnsatisfiedLinkError:HelloNative.sayHello()V\x0d\x0aatHelloNative.sayHello(NativeMethod)\x0d\x0aatTestNative.main(TestNative.java:13)\x0d\x0a\x0d\x0a載入庫(kù)失敗,但是庫(kù)明明就是放在當(dāng)前文件夾下的,怎么會(huì)載入失敗呢?\x0d\x0a用System.getProperty("java.library.path")查看,發(fā)現(xiàn)java.library.path中并不u存在當(dāng)前的目錄.主要有以下的幾個(gè)解決辦法:\x0d\x0a1)將生成的庫(kù)復(fù)制到j(luò)ava.library.path有的路徑中去,當(dāng)然這樣不是很好\x0d\x0a2)設(shè)置環(huán)境變量exportLD_LIBRARY_PATH=.:$LD_LIBRARY_PATH,將當(dāng)前的目錄加入到LD_LIBRARY_PATH中\(zhòng)x0d\x0a3)設(shè)置java的選項(xiàng),將當(dāng)前的目錄加入到其中.java-Djava.library.path=.$LD_LIBRARY_PATH\x0d\x0a這樣之后程序就能夠成功的運(yùn)行了.可以看見(jiàn)顯示的"HelloWorld!"了
為泗洪等地區(qū)用戶提供了全套網(wǎng)頁(yè)設(shè)計(jì)制作服務(wù),及泗洪網(wǎng)站建設(shè)行業(yè)解決方案。主營(yíng)業(yè)務(wù)為網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站制作、泗洪網(wǎng)站設(shè)計(jì),以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠(chéng)的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會(huì)得到認(rèn)可,從而選擇與我們長(zhǎng)期合作。這樣,我們也可以走得更遠(yuǎn)!
Java調(diào)用C語(yǔ)言程序時(shí),主要是涉及到操作系統(tǒng)底層的事件。這種時(shí)間Java無(wú)法處理,例如用戶上傳一個(gè)視頻文件,需要后臺(tái)給視頻加上水印,或者后臺(tái)分離視頻流和音頻流。只能通過(guò)調(diào)用C語(yǔ)言處理。
使用Java如何去調(diào)用C語(yǔ)言的接口呢?使用Java的JNI技術(shù)。
具體調(diào)用步驟如下:
1.首先創(chuàng)建Java文件 HelloJni.java ,并創(chuàng)建native方法。
2.編譯Java文件并生成java頭文件。
3.創(chuàng)建C語(yǔ)言文件,HelloWorld.c。
4.生成動(dòng)態(tài)鏈接庫(kù)文件 libhello.so。
5.設(shè)置動(dòng)態(tài)鏈接庫(kù)文件的目錄。
6.把剛才生成的so文件拷貝到/home/lib下,然后執(zhí)行class文件。
JAVA以其跨平臺(tái)的特性深受人們喜愛(ài),而又正由于它的跨平臺(tái)的目的,使得它和本地機(jī)器的各種內(nèi)部聯(lián)系變得很少,約束了它的功能。解決JAVA對(duì)本地操作的一種方法就是JNI。
JAVA通過(guò)JNI調(diào)用本地方法,而本地方法是以庫(kù)文件的形式存放的(在WINDOWS平臺(tái)上是DLL文件形式,在UNIX機(jī)器上是SO文件形式)。通過(guò)調(diào)用本地的庫(kù)文件的內(nèi)部方法,使JAVA可以實(shí)現(xiàn)和本地機(jī)器的緊密聯(lián)系,調(diào)用系統(tǒng)級(jí)的各接口方法。
簡(jiǎn)單介紹及應(yīng)用如下:
一、JAVA中所需要做的工作
在JAVA程序中,首先需要在類中聲明所調(diào)用的庫(kù)名稱,如下:
static
{
System.loadLibrary(“goodluck”);
}
在這里,庫(kù)的擴(kuò)展名字可以不用寫(xiě)出來(lái),究竟是DLL還是SO,由系統(tǒng)自己判斷。
還需要對(duì)將要調(diào)用的方法做本地聲明,關(guān)鍵字為native。并且只需要聲明,而不需要具體實(shí)現(xiàn)。如下:
public
native
static
void
set(int
i);
public
native
static
int
get();
然后編譯該JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就會(huì)生成C/C++的頭文件。
例如程序testdll.java,內(nèi)容為:
public
class
testdll
{
static
{
System.loadLibrary("goodluck");
}
public
native
static
int
get();
public
native
static
void
set(int
i);
public
static
void
main(String[]
args)
{
testdll
test
=
new
testdll();
test.set(10);
System.out.println(test.get());
}
}
用javac
testdll.java編譯它,會(huì)生成testdll.class。
再用javah
testdll,則會(huì)在當(dāng)前目錄下生成testdll.h文件,這個(gè)文件需要被C/C++程序調(diào)用來(lái)生成所需的庫(kù)文件。
二、C/C++中所需要做的工作
對(duì)于已生成的.h頭文件,C/C++所需要做的,就是把它的各個(gè)方法具體的實(shí)現(xiàn)。然后編譯連接成庫(kù)文件即可。再把庫(kù)文件拷貝到JAVA程序的路徑下面,就可以用JAVA調(diào)用C/C++所實(shí)現(xiàn)的功能了。
接上例子。我們先看一下testdll.h文件的內(nèi)容:
/*
DO
NOT
EDIT
THIS
FILE
-
it
is
machine
generated
*/
#include
jni.h
/*
Header
for
class
testdll
*/
#ifndef
_Included_testdll
#define
_Included_testdll
#ifdef
__cplusplus
extern
"C"
{
#endif
/*
*
Class:
testdll
*
Method:
get
*
Signature:
()I
*/
JNIEXPORT
jint
JNICALL
Java_testdll_get
(JNIEnv
*,
jclass);
/*
*
Class:
testdll
*
Method:
set
*
Signature:
(I)V
*/
JNIEXPORT
void
JNICALL
Java_testdll_set
(JNIEnv
*,
jclass,
jint);
#ifdef
__cplusplus
}
#endif
#endif
在具體實(shí)現(xiàn)的時(shí)候,我們只關(guān)心兩個(gè)函數(shù)原型
JNIEXPORT
jint
JNICALL
Java_testdll_get
(JNIEnv
*,
jclass);
和
JNIEXPORT
void
JNICALL
Java_testdll_set
(JNIEnv
*,
jclass,
jint);
這里JNIEXPORT和JNICALL都是JNI的關(guān)鍵字,表示此函數(shù)是要被JNI調(diào)用的。而jint是以JNI為中介使JAVA的int類型與本地的int溝通的一種類型,我們可以視而不見(jiàn),就當(dāng)做int使用。函數(shù)的名稱是JAVA_再加上java程序的package路徑再加函數(shù)名組成的。參數(shù)中,我們也只需要關(guān)心在JAVA程序中存在的參數(shù),至于JNIEnv*和jclass我們一般沒(méi)有必要去碰它。
好,下面我們用testdll.cpp文件具體實(shí)現(xiàn)這兩個(gè)函數(shù):
#include
"testdll.h"
int
i
=
0;
JNIEXPORT
jint
JNICALL
Java_testdll_get
(JNIEnv
*,
jclass)
{
return
i;
}
JNIEXPORT
void
JNICALL
Java_testdll_set
(JNIEnv
*,
jclass,
jint
j)
{
i
=
j;
}
編譯連接成庫(kù)文件,本例是在WINDOWS下做的,生成的是DLL文件。并且名稱要與JAVA中需要調(diào)用的一致,這里就是goodluck.dll
把goodluck.dll拷貝到testdll.class的目錄下,java
testdll運(yùn)行它,就可以觀察到結(jié)果了。