JAVA以其跨平臺的特性深受人們喜愛,而又正由于它的跨平臺的目的,使得它和本地機(jī)器的各種內(nèi)部聯(lián)系變得很少,約束了它的功能。解決JAVA對本地操作的一種方法就是JNI。
成都創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括如皋網(wǎng)站建設(shè)、如皋網(wǎng)站制作、如皋網(wǎng)頁制作以及如皋網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,如皋網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到如皋省份的部分城市,未來相信會繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
JAVA通過JNI調(diào)用本地方法,而本地方法是以庫文件的形式存放的(在WINDOWS平臺上是DLL文件形式,在UNIX機(jī)器上是SO文件形式)。通過調(diào)用本地的庫文件的內(nèi)部方法,使JAVA可以實(shí)現(xiàn)和本地機(jī)器的緊密聯(lián)系,調(diào)用系統(tǒng)級的各接口方法。
簡單介紹及應(yīng)用如下:
一、JAVA中所需要做的工作
在JAVA程序中,首先需要在類中聲明所調(diào)用的庫名稱,如下:
static {
System.loadLibrary(“goodluck”);
}
在這里,庫的擴(kuò)展名字可以不用寫出來,究竟是DLL還是SO,由系統(tǒng)自己判斷。
還需要對將要調(diào)用的方法做本地聲明,關(guān)鍵字為native。并且只需要聲明,而不需要具體實(shí)現(xiàn)。如下:
public native static void set(int i);
public native static int get();
然后編譯該JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就會生成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編譯它,會生成testdll.class。
再用javah testdll,則會在當(dāng)前目錄下生成testdll.h文件,這個(gè)文件需要被C/C++程序調(diào)用來生成所需的庫文件。
二、C/C++中所需要做的工作
對于已生成的.h頭文件,C/C++所需要做的,就是把它的各個(gè)方法具體的實(shí)現(xiàn)。然后編譯連接成庫文件即可。再把庫文件拷貝到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;
}
編譯連接成庫文件,本例是在WINDOWS下做的,生成的是DLL文件。并且名稱要與JAVA中需要調(diào)用的一致,這里就是goodluck.dll
把goodluck.dll拷貝到testdll.class的目錄下,java testdll運(yùn)行它,就可以觀察到結(jié)果了。
先把2個(gè)文件放在同一個(gè)包下面,這樣就可以訪問另一個(gè)java程序了。
一般的方法是在A程序里實(shí)例化B類,然后通過B.方法名 去調(diào)用B類里的方法
Runtime.getRuntime().exec("外部程序");
相當(dāng)于你在cmd控制臺中輸入"外部程序"并回車執(zhí)行
public class HelloWord{
//自定義的輸出函數(shù),打印字符串helloword!
public void printHelloWord() {
System.out.println("helloword!");
}
public static void main(String args[]){
//創(chuàng)建對象實(shí)例
HelloWord hw = new HelloWord();
//通過對象實(shí)例調(diào)用自己的成員函數(shù)
hw.printHelloWord() ;
}
}
你看看滿意不?
服務(wù)器:
import?java.io.*;
import?java.net.*;
import?java.util.*;
public?class?TalkServer{
public?static?void?main(String[]?args){
try{
ServerSocket?server?=?new?ServerSocket(6666);
Socket?socket?=?server.accept();
String?line?=?null;
BufferedReader?br?=?new?BufferedReader(new?InputStreamReader(socket.getInputStream()));
PrintWriter?pw?=?new?PrintWriter(socket.getOutputStream());
BufferedReader?br2?=?new?BufferedReader(new?InputStreamReader(System.in));
System.out.println("Client:"?+?br.readLine());
line?=?br2.readLine();
while(!line.equals("bye")){
pw.println(line);
pw.flush();
System.out.println("Server:"?+?line);
System.out.println("Client:"?+?br.readLine());
line?=?br2.readLine();
}
br.close();
pw.close();
socket.close();
server.close();
}catch(Exception?e){
System.out.println("Error:"?+?e);
}
}
}
客戶端:
import?java.io.*;
import?java.net.*;
import?java.util.*;
public?class?TalkClient{
public?static?void?main(String[]?args){
try{
Socket?socket?=?new?Socket("127.0.0.1",6666);
BufferedReader?br?=?new?BufferedReader(new?InputStreamReader(System.in));
PrintWriter?pw?=?new?PrintWriter(socket.getOutputStream());
BufferedReader?br2?=?new?BufferedReader(new?InputStreamReader(socket.getInputStream()));
String?readLine?=?null;
readLine?=?br.readLine();
while(!readLine.equals("bye")){
pw.println(readLine);
pw.flush();
System.out.println("Client:"?+?readLine);
System.out.println("Server:"?+?br2.readLine());
readLine?=?br.readLine();
}
br.close();
pw.close();
socket.close();
}catch(Exception?e){
System.out.println("Client?Error:"?+?e);
}
}
}