java jni調(diào)用c函數(shù)實(shí)例分享(java調(diào)用c函數(shù))
從C/C++到Java,再?gòu)腏ava回到C/C++,今天終于有機(jī)會(huì)了解了連接Java、C/C++的橋梁——JNI。哈哈!分享一下!
一、簡(jiǎn)介
JNI是Java native interface的簡(jiǎn)寫,可以譯作Java原生接口。Java可以通過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. 編寫并編譯Java程序
javac J2C.java => J2C.class
2. 生成C/C++頭文件
javah J2C => J2C.h (安裝JDK后,$JAVA_HOME應(yīng)該已加入$PATH, 否則使用絕對(duì)路徑,例如/usr/bin/javah)
3. 編寫對(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
五、具體過程
1. 編寫并編譯J2C.java
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)程(即主線程)的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開頭)
2. 生成C頭文件J2C.h:javah J2C
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* 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);
按照注釋的說明是在J2C.java文件的類J2C的方法write2proc處定義,故C程序的實(shí)現(xiàn)函數(shù)必須與該處簽名一致;
3. 編寫C程序J2C.c
#include <stdio.h>
#include "J2C.h"
JNIEXPORT int JNICALL Java_J2C_write2proc(JNIEnv * env, jobject arg, jint pid)
{
printf("current pid is %d\n", pid);
return 0;
}
4. 編譯C程序
因?yàn)镃程序里#include "J2C.h"而J2C.h又#include <jni.h>, 而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)文件定義必須一致(可以通過自動(dòng)生成的C/C++頭文件來(lái)比較),尤其是類名和方法名;
2. Java中System.loadLibrary()載入的共享庫(kù)名必須與后面C/C++生成的共享庫(kù)名一致。
相關(guān)文章
SpringBoot項(xiàng)目使用?axis?調(diào)用webservice接口的實(shí)踐記錄
這篇文章主要介紹了SpringBoot項(xiàng)目使用?axis?調(diào)用webservice接口,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06解決fcitx輸入法在IDEA中輸入法候選框無(wú)法跟隨光標(biāo)移動(dòng)的問題
這篇文章主要介紹了解決fcitx輸入法在Intellij IDEA開發(fā)工具中輸入法候選框無(wú)法跟隨光標(biāo)移動(dòng)的問題,代碼簡(jiǎn)單易懂對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10Java Socket編程心跳包創(chuàng)建實(shí)例解析
這篇文章主要介紹了Java Socket編程心跳包創(chuàng)建實(shí)例解析,具有一定借鑒價(jià)值,需要的朋友可以參考下2017-12-12mybatis配置對(duì)象包含對(duì)象以及List的方式
這篇文章主要介紹了mybatis配置對(duì)象包含對(duì)象以及List的方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06spring?boot教程之建立第一個(gè)HelloWorld
這篇文章主要介紹了spring?boot教程之建立第一個(gè)HelloWorld的相關(guān)資料,需要的朋友可以參考下2022-08-08