JAVA以其跨平臺(tái)的特性深受人們喜愛,而又正由于它的跨平臺(tái)的目的,使得它和本地機(jī)器的各種內(nèi)部聯(lián)系變得很少,約束了它的功能。解決JAVA對(duì)本地操作的一種方法就是JNI。
10年積累的成都網(wǎng)站設(shè)計(jì)、做網(wǎng)站經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)制作后付款的網(wǎng)站建設(shè)流程,更有點(diǎn)軍免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
JAVA通過JNI調(diào)用本地方法,而本地方法是以庫(kù)文件的形式存放的(在WINDOWS平臺(tái)上是DLL文件形式,在UNIX機(jī)器上是SO文件形式)。通過調(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ò)展名字可以不用寫出來,究竟是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)用來生成所需的庫(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溝通的一種類型,我們可以視而不見,就當(dāng)做int使用。函數(shù)的名稱是JAVA_再加上java程序的package路徑再加函數(shù)名組成的。參數(shù)中,我們也只需要關(guān)心在JAVA程序中存在的參數(shù),至于JNIEnv*和jclass我們一般沒有必要去碰它。
好,下面我們用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é)果了。
要在java中調(diào)用c語言的庫(kù),需要使用Java提供了JNI。
舉例說明
在c語言中定義一個(gè) void sayHello()函數(shù)(打印Hello World);然后在Java中調(diào)用這個(gè)函數(shù)顯示Hello Word.
現(xiàn)在分別從Java和C語言兩部分說明:
1. Java 部分
首先定義一個(gè)HelloNative,在其中申明sayHello函數(shù),函數(shù)要申明為Native 類型的.如下:
public class HelloNative {
public native void sayHello();
}
編譯這個(gè)類,生成class文件:
javac HelloWorld.java
利用javah生成需要的h文件
javah HelloNative
生成的 h文件大概如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include jni.h
/* Header for class HelloNative */
#ifndef _Included_HelloNative
#define _Included_HelloNative
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloNative
* Method: sayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloNative_sayHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
可以看一下上面自動(dòng)生成的程序,程序include了jni.h,這個(gè)頭文件在 $JAVA_HOME下的include文件夾下. 還可以發(fā)現(xiàn)生成的函數(shù)名是在之前的函數(shù)名前面加上了Java_HelloNative。
2. C語言部分
根據(jù)上面生成的h文件編寫相應(yīng)的代碼實(shí)現(xiàn),建立一個(gè) HelloNative.cpp用來實(shí)現(xiàn)顯示Hello World的函數(shù).如下:
#include stdio.h
#include "HelloNative.h"
JNIEXPORT void JNICALL Java_HelloNative_sayHello(JNIEnv *, jobject)
{
printf("Hello World!\n");
}
代碼編寫完成之后,我們?cè)儆胓cc編譯成庫(kù)文件,命令如下;
gcc -fPIC -I/usr/lib/jvm/java-7-openjdk-i386/include -I/usr/lib/jvm/java-7-openjdk-i386/include/linux -shared -o libHelloNative.so HelloNative.cpp
這樣就會(huì)在當(dāng)前目錄下生成一個(gè)libHelloNative.so的庫(kù)文件.這時(shí)需要的庫(kù)已經(jīng)生成,在C語言下的工作已經(jīng)完成了.
接下來需要在Java中編寫一個(gè)程序測(cè)試一下.在程序前,需要將我們的庫(kù)載入進(jìn)去.載入的方法是調(diào)用Java的 System.loadLibrary("HelloNative");
public class TestNative
{
static {
try {
System.loadLibrary("HelloNative");
}
catch(UnsatisfiedLinkError e) {
System.out.println( "Cannot load hello library:\n " + e.toString() );
}
}
public static void main(String[] args) {
HelloNative test = new HelloNative();
test.sayHello();
}
}
但是再編譯后,運(yùn)行的時(shí)候,問題又出現(xiàn)了.
Cannot load hello library:
java.lang.UnsatisfiedLinkError: no HelloNative in java.library.path
Exception in thread "main" java.lang.UnsatisfiedLinkError: HelloNative.sayHello()V
at HelloNative.sayHello(Native Method)
at TestNative.main(TestNative.java:13)
載入庫(kù)失敗,但是庫(kù)明明就是放在當(dāng)前文件夾下的,怎么會(huì)載入失敗呢?
用System.getProperty("java.library.path")查看,發(fā)現(xiàn)java.library.path中并不u存在當(dāng)前的目錄.主要有以下的幾個(gè)解決辦法:
1) 將生成的庫(kù)復(fù)制到j(luò)ava.library.path有的路徑中去,當(dāng)然這樣不是很好
2) 設(shè)置環(huán)境變量export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH ,將當(dāng)前的目錄加入到LD_LIBRARY_PATH中
3) 設(shè)置java 的選項(xiàng),將當(dāng)前的目錄加入到其中 .java -Djava.library.path=. $LD_LIBRARY_PATH
這樣之后程序就能夠成功的運(yùn)行了.可以看見顯示的"Hello World!"了
用 Runtime 的 exec 方法的確是可行的。
假設(shè)我們已經(jīng)把以下的 C 程序編繹成 adder.exe:
#include stdio.h
int main() { /* 簡(jiǎn)單地循環(huán)打印標(biāo)準(zhǔn)輸入上的兩個(gè)整數(shù)之和 */
int a, b, lineNumber = 0;
while (scanf("%d %d", a, b))
printf("Line# %d \t %d + %d == %d\n", ++lineNumber, a, b, a + b);
return 0;
}
以下的 Java 程序可以在啟動(dòng) adder.exe 后,跟 adder.exe 的標(biāo)準(zhǔn)輸入和輸出接軌,然后持續(xù)不斷地向它發(fā)送數(shù)據(jù)和索取結(jié)果:
import java.io.*;
class C {
public static void main(String[] args) throws Exception {
final Process proc = Runtime.getRuntime().exec("adder.exe");
// 用另一個(gè)線程把參數(shù)送到 proc 的標(biāo)準(zhǔn)輸入上去。
new Thread() {
public void run() {
OutputStream stdin = proc.getOutputStream();
for (int i = 0; ; i++) {
try {
Thread.sleep(1); // 要休息片刻才看得到 I/O 的緩存效果。
stdin.write((i + " " + i + "\n").getBytes());
} catch (Exception ex) {
ex.printStackTrace();
}
}
}
}.start();
// 主線程負(fù)責(zé)讀取并打印 proc 的標(biāo)準(zhǔn)輸出。
BufferedReader stdout = new BufferedReader(new InputStreamReader(proc.getInputStream()));
for (String line; null != (line = stdout.readLine()); )
System.out.println(line);
}
}
循環(huán)里的 Thread.sleep(1) 純粹是為了凸顯 I/O 的緩存效果。
我測(cè)試時(shí)看到大約 900 行的緩存量(用 32-bit XP 和 Java 1.6)。
我知道的有兩種方法,一種是直接用JNI,另一種是則是用Jacob(實(shí)質(zhì)上也用的是JNI)
先講講第一種方法
1.編寫Java程序TestDll,注意,這個(gè)類有兩個(gè)作用,一個(gè)是用來做頭文件,另外一個(gè)作用就是通過它來調(diào)用dll
public class TestDll {
static
{
System.loadLibrary("DLLSample");//載入dll
}
public native static int DoubleValue(int i);//函數(shù)聲明
}
2.編譯: javac TestDll
3.生成頭文件: javah TestDll
生成TestDll.h文件,這里面只對(duì)函數(shù)DoubleValue作了聲明
/* 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: DoubleValue
* Signature: (I)I
*/
JNIEXPORT jint JNICALL Java_TestDll_DoubleValue
(JNIEnv *, jclass, jint);
#ifdef __cplusplus
}
#endif
#endif
4.用VC制作DLL
修改DLLSample工程,程序中添加函數(shù)DoubleValue的實(shí)現(xiàn),函數(shù)名必須用jni規(guī)定格式,可以照到.h文件里的聲
明來寫:
#include "jni_md.h"
#include "TestDll.h"
JNIEXPORT jint JNICALL Java_TestDll_DoubleValue (JNIEnv *, jclass, jint p)
{
int j = p*2;
return j;
}
JNIEXPORT,JNICALL,JNIEnv *, jclass請(qǐng)都不要?jiǎng)樱琷int對(duì)應(yīng)是java里的int
別忘了把TestDll.h,jni.h(這個(gè)文件在JDK/include下),jni_md.h(這個(gè)文件在jdk/include/win32下)復(fù)制到VC
工程目錄中,然后編譯生成dll
5.把生成的DLLSample.dll復(fù)制到j(luò)dk/bin下(確保PATH指向了這個(gè)目錄)
6.應(yīng)用
我看到他們舉的例子都是把調(diào)用寫在了main里, 在這里我把應(yīng)用取出來放在一個(gè)Servlet類中的測(cè)試的,同樣可
以得到正確結(jié)果,這里體現(xiàn)國(guó)TestDll的第二個(gè)作用,它是調(diào)用dll的一個(gè)紐帶
......
TestDll td = new TestDll();
System.out.println("result= " td.DoubleValue(25));
.....
注意:
如果更改了TestDll的內(nèi)容或重使命名了這個(gè)文件,必須重新用它生成頭文件,在dll里也要做相應(yīng)修改,
因?yàn)楸仨氁蚸ava里面的聲明一致才行
----------------------------------------
本文隨寫,若有什么錯(cuò)誤,請(qǐng)多多指出,謝謝
說明:本文是west263的博文,作者并沒有留名
在java程序中,我們可以通過JNI實(shí)現(xiàn)一些用java語言不便實(shí)現(xiàn)的功能。通常有以下幾種情況我們需要使用JNI來實(shí)現(xiàn)。 標(biāo)準(zhǔn)的java類庫(kù)沒有提供你的應(yīng)用程序所需要的功能,通常這些功能是平臺(tái)相關(guān)的 你希望使用一些已經(jīng)有的類庫(kù)或者應(yīng)用程序,而他們并非用java語言編寫的 程序的某些部分對(duì)速度要求比較苛刻,你選擇用匯編或者c語言來實(shí)現(xiàn)并在java語言中調(diào)用他們 下面我們開始編寫HelloWorld程序,由于涉及到要編寫c/c++代碼因此我們會(huì)在開發(fā)中使用Microsoft VC++工具。編寫java代碼,我們?cè)谟脖P上建立一個(gè)hello目錄作為我們的工作目錄,首先我們需要編寫自己的java代碼,在java代碼中我們會(huì)聲明native方法,代碼非常簡(jiǎn)單。如下所示 class HelloWorld { public native void displayHelloWorld(); static { System.loadLibrary("hello"); } public static void main(String[] args) { new HelloWorld().displayHelloWorld(); } } 注意我們的displayHelloWorld()方法的聲明,它有一個(gè)關(guān)鍵字native,表明這個(gè)方法使用java以外的語言實(shí)現(xiàn)。方法不包括實(shí)現(xiàn),因?yàn)槲覀円胏/c++語言實(shí)現(xiàn)它。注意System.loadLibrary("hello")這句代碼,它是在靜態(tài)初始化塊中定義的,系統(tǒng)用來裝載hello共享庫(kù),這就是我們?cè)诤竺嫔傻膆ello.dll(如果在其他的操作系統(tǒng)可能是其他的形式,比如hello.so) 編譯java代碼 javac HelloWorld.java 生成HelloWorld.class文件 創(chuàng)建.h文件 這一步中我們要使用javah命令生成.h文件,這個(gè)文件要在后面的c/c++代碼中用到,我們運(yùn)行 javah HelloWorld。這樣我們可以看到在相同目錄下生成了一個(gè)HelloWorld.h文件,文件內(nèi)容如下 在此我們不對(duì)他進(jìn)行太多的解釋。 /* DO NOT EDIT THIS FILE - it is machine generated */ #include /* Header for class HelloWorld */ #ifndef _Included_HelloWorld #define _Included_HelloWorld #ifdef __cplusplus extern "C" { #endif /* * Class: HelloWorld * Method: displayHelloWorld * Signature: ()V */ JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld (JNIEnv *, jobject); #ifdef __cplusplus } #endif #endif 編寫本地實(shí)現(xiàn)代碼 在這部分我們要用C/C++語言實(shí)現(xiàn)java中定義的方法,我們?cè)赩C++中新建一個(gè)Project,然后創(chuàng)建一個(gè)HelloWorldImp.cpp文件,內(nèi)容如下 #include #include "HelloWorld.h" #include JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld(JNIEnv *env, jobject obj) { printf("Hello world!\n"); return; } 注意我們這里include了 jni.h和剛才得到的HelloWorld.h文件。因此你要在VC++里面設(shè)置好,jni.h在JAVA_HOME/include里面。編譯通過后再生成hello.dll文件。 運(yùn)行java程序 把上面生成的hello.dll文件復(fù)制到我們的工作目錄,這時(shí)候我們的目錄中包括HelloWorld.java,HelloWorld.class和hello.dll文件。運(yùn)行java HelloWorld命令,則可在控制臺(tái)看到Hello world| 的輸出了。運(yùn)行VC++; 文件---新建---選“win32 console application”(控制臺(tái)程序)---在右方設(shè)置好路徑并輸入工程名---確定 接下來的幾個(gè)提示框點(diǎn)確定就行了,那是提示是否要用VC++提供的框架之類的 在左邊的工作空間中選“FILEVIEW”標(biāo)簽項(xiàng),點(diǎn)開“+”號(hào),右鍵點(diǎn)擊“SOURCE FILES”,選“添加文件到目錄”,此即添加你要建立的C++源程序文件,會(huì)提示你沒有文件,是否添加,你點(diǎn)是,輸入文件名保存就OK了 然后SOURCE FILES下就出現(xiàn)了你剛才建立的*.CPP文件,雙擊,輸入代碼. 以下就是點(diǎn)”組建”菜單中的”編譯”、”組建”等命令進(jìn)行調(diào)試了。 相信你會(huì)了。
當(dāng)前文章:java中調(diào)用c代碼 java調(diào)用c語言
網(wǎng)站URL:http://vcdvsql.cn/article28/doodejp.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站建設(shè)、品牌網(wǎng)站設(shè)計(jì)、服務(wù)器托管、網(wǎng)站改版、定制網(wǎng)站、手機(jī)網(wǎng)站建設(shè)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)