- 相關(guān)推薦
java調(diào)用c函數(shù)的實(shí)例
從C/C++到Java,再?gòu)腏ava回到C/C++,今天終于有機(jī)會(huì)了解了連接Java、C/C++的橋梁——JNI。哈哈!分享一下!
一、簡(jiǎn)介
JNI是Java native interface的簡(jiǎn)寫(xiě),可以譯作Java原生接口。Java可以通過(guò)JNI調(diào)用C/C++的庫(kù),這對(duì)于那些對(duì)性能要求比較高的Java程序或者Java無(wú)法處理的任務(wù)無(wú)疑是一個(gè)很好的方式。
二、目的:Java代碼中調(diào)用C/C++代碼
三、實(shí)現(xiàn):假設(shè)我們的Java程序?yàn)镴2C.java, C程序?yàn)镴2C.c, Java與C之間的通信函數(shù)名為write2proc;
那么write2proc的聲明位于J2C.java,實(shí)現(xiàn)位于J2C.c;
四、操作
1. 編寫(xiě)并編譯Java程序
javac J2C.java => J2C.class
2. 生成C/C++頭文件
javah J2C => J2C.h (安裝JDK后,$JAVA_HOME應(yīng)該已加入$PATH, 否則使用絕對(duì)路徑,例如/usr/bin/javah)
3. 編寫(xiě)對(duì)應(yīng)的C/C++程序:J2C.c
4. 生成C/C++目標(biāo)文件
gcc -I/usr/lib/jvm/java-6-openjdk-amd64/include -I/usr/lib/jvm/java-6-openjdk-amd64/include/linux -fPIC -c J2C.c => J2C.o
5. 生成C/C++共享庫(kù)
gcc -shared -Wl,-soname,libj2c.so.1 -o libj2c.so.1.0 J2C.o => libj2c.so.1.0
6. 重命名cp libj2c.so.1.0 libj2c.so => libj2c.so
7. 將共享庫(kù)加入動(dòng)態(tài)鏈接庫(kù)的路徑(此例為當(dāng)前目錄)
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:.
8. 執(zhí)行Java程序,實(shí)現(xiàn)跨語(yǔ)言通信
java J2C
五、具體過(guò)程
1. 編寫(xiě)并編譯J2C.java
復(fù)制代碼 代碼如下:
import java.lang.management.ManagementFactory;
import java.lang.management.RuntimeMXBean;
public class J2C
{
static
{
try{
// 此處即為本地方法所在鏈接庫(kù)名
System.loadLibrary("j2c");
} catch(UnsatisfiedLinkError e)
{
System.err.println( "Cannot load J2C library:n " +
e.toString() );
}
}
//聲明的本地方法
public static native int write2proc(int pid);
public static void main(String[] args){
//獲取本進(jìn)程(即主線(xiàn)程)的pid
final RuntimeMXBean runtime = ManagementFactory.getRuntimeMXBean();
final String info = runtime.getName();
final int index = info.indexOf("@");
if (index != -1) {
final int pid = Integer.parseInt(info.substring(0, index));
System.out.println(info);
System.out.println(pid);
write2proc(pid);
}
try{
Thread.sleep(8000);
} catch(InterruptedException e){
e.printStackTrace();
}
}
}
note:Java程序中System.loadLibrary參數(shù)名表示要載入的C/C++共享庫(kù),第6步生成的共享庫(kù)名必須與該參數(shù)一致,即System.loadLibrary(Name) 對(duì)應(yīng)共享庫(kù)名libName.so (共享庫(kù)名必須以lib開(kāi)頭)
2. 生成C頭文件J2C.h:javah J2C
復(fù)制代碼 代碼如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include
/* Header for class J2C */
#ifndef _Included_J2C
#define _Included_J2C
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: J2C
* Method: write2proc
* Signature: (I)I
*/
JNIEXPORT jint JNICALL Java_J2C_write2proc
(JNIEnv *, jclass, jint);
#ifdef __cplusplus
}
#endif
#endif
note:1. 頭文件自動(dòng)生成,不要修改它;
2. 函數(shù)JNIEXPORT jint JNICALL Java_J2C_write2proc(JNIEnv *, jclass, jint);
按照注釋的說(shuō)明是在J2C.java文件的類(lèi)J2C的方法write2proc處定義,故C程序的實(shí)現(xiàn)函數(shù)必須與該處簽名一致;
3. 編寫(xiě)C程序J2C.c
復(fù)制代碼 代碼如下:
#include
#include "J2C.h"
JNIEXPORT int JNICALL Java_J2C_write2proc(JNIEnv * env, jobject arg, jint pid)
{
printf("current pid is %dn", pid);
return 0;
}
4. 編譯C程序
因?yàn)镃程序里#include "J2C.h"而J2C.h又#include, 而gcc里面默認(rèn)環(huán)境并不知道jni.h是什么東西,故編譯時(shí)需要告訴編譯器jni.h的位置( jni.h在jdk 的$JAVA_HOME/include下面),所以才有了上面的編譯參數(shù);
因?yàn)槭褂胓cc編譯得到動(dòng)態(tài)庫(kù),在jni調(diào)用的時(shí)候,某些情況會(huì)有異常, 可嘗試改用g++。
總結(jié)
1. Java中方法的原型聲明與C/C++對(duì)應(yīng)的實(shí)現(xiàn)文件定義必須一致(可以通過(guò)自動(dòng)生成的C/C++頭文件來(lái)比較),尤其是類(lèi)名和方法名;
2. Java中System.loadLibrary()載入的共享庫(kù)名必須與后面C/C++生成的共享庫(kù)名一致。
【java調(diào)用c函數(shù)的實(shí)例】相關(guān)文章:
C語(yǔ)言中Swift函數(shù)調(diào)用實(shí)例09-25
Java程序調(diào)用C/C++語(yǔ)言函數(shù)的方法07-31
Java程序如何調(diào)用C/C++語(yǔ)言函數(shù)10-29
匯編調(diào)用C函數(shù)10-29
php函數(shù)連續(xù)調(diào)用實(shí)例分析07-30