最近正在做一個(gè)C/C++調(diào)用java的程序,這里說的調(diào)用java不是使用方式
exec(/path/to/java,.....),而是調(diào)用一個(gè)class文件中的一個(gè)特定的函數(shù)。
實(shí)踐后總結(jié)如下:
1. 安裝 jdk
2. 安裝gcc(linux自帶有的就無需安裝了)
利用JNI(java native interface),來實(shí)現(xiàn)動(dòng)態(tài)建立java runtime
environment.
第一,C/C++程序中包含頭文件"jni.h"
#include <jni.h>
一般在JAVA_HOME/include 目錄下。
調(diào)用jni.h中的方法建立runtime env 然后調(diào)用java 程序。
第二,編譯
g++ -o testjava testjava.cpp -I${JAVA_HOME}/include
-I${JAVA_HOME}/include/linux -L${JRE_HOME}/lib/i386/client
-ljvm
以上就是大致思路,現(xiàn)詳細(xì)說明過程如下:
#####################################################################################
一、安裝配置Java環(huán)境
我的linux是RedHat Enterprise linux 5, 內(nèi)核版本2.6.18
在Linux系統(tǒng)中安裝Java比較簡(jiǎn)單??梢栽L問Java
download網(wǎng)站或自由軟件庫(kù)等,選擇你所有安裝的操作系統(tǒng)類型(Linux,Linux
AMD64,Solaris等)。一旦你已經(jīng)選擇下載文件──要么是自解壓縮執(zhí)行文件,要么是自解壓縮的RPM文件,你都可以安裝它。我下載的是jdk-1_5_0_06-linux-i586.bin:
# mkdir /usr/local/java
# cd /usr/local/java
# cp /home/soft/jdk-1_5_0_06-linux-i586.bin ./
# chmod u+x jdk-1_5_0_06-linux-i586.bin
# ./jdk-1_5_0_06-linux-i586.bin
|
運(yùn)行完后生成jdk1.5.0_06目錄,jdk被安裝在/usr/local/java/jdk1.5.0_06/。運(yùn)行以下執(zhí)行代碼將得到一個(gè)測(cè)試結(jié)果:
# cd jdk1.5.0_06/bin
[root@localhost bin]# ./java -version
java version "1.5.0_06"
Java(TM) 2 Runtime Environment, Standard Edition
(build 1.5.0_06-b05)
Java HotSpot(TM) Client VM (build 1.5.0_06-b05,
mixed mode, sharing)
|
為了能夠使用Java,需要設(shè)置如下環(huán)境變量:
JAVA_HOME=/usr/local/java/jdk1.5.0_06
PATH=$PATH:/usr/local/java/jre1.5.0_05/bin
export JAVA_HOME PATH
export
JRE_HOME=/usr/lib/jvm/java-1.4.2-gcj-1.4.2.0/jre
export
LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$JRE_HOME/lib/i386:$JRE_HOME/lib/i386/client
|
注意JRE_HOME的配置,若機(jī)器上沒有jre環(huán)境,則安裝jre,安裝方法類似安裝jdk
設(shè)置完后可以查看變量的值
[root@localhost bin]# echo $JAVA_HOME
/usr/local/java/jdk1.5.0_06
[root@localhost bin]# echo $PATH
/usr/kerberos/sbin:/usr/kerberos/bin:/usr/local/bin:/bin:/usr/bin:/home/zhangp/bin:/usr/lib/jvm/java-1.4.2-gcj-1.4.2.0/jre/bin:/usr/local/java/jdk1.5.0_06/bin
二、編寫簡(jiǎn)單的Java程序
package com.test;
public class MyTest {
public MyTest(){
super();
}
public static int add(int a,int b) {
return a+b;
}
public boolean judge(boolean bool) {
return
!bool;
}
}
編譯Java程序:
#javac MyTest.java
編譯之后生成MyTest.class,將其放置于當(dāng)前目錄的com/test目錄下,C++程序的JNI調(diào)用時(shí)會(huì)使用相關(guān)方法在com/test目錄下查找該class。
三、C++程序
#include <stdio.h>
#include <iostream>
#include <jni.h>
#include <stdlib.h>
#include <assert.h>
jstring stoJstring(JNIEnv* env, const char* pat)
{
jclass strClass =
env->FindClass("java/lang/String");
jmethodID ctorID =
env->GetMethodID(strClass,
"<init>",
"([BLjava/lang/String;)V");
jbyteArray bytes =
env->NewByteArray(strlen(pat));
env->SetByteArrayRegion(bytes, 0,
strlen(pat), (jbyte*)pat);
jstring encoding =
env->NewStringUTF("utf-8");
return
(jstring)env->NewObject(strClass, ctorID, bytes,
encoding);
}
char* jstringTostring(JNIEnv* env, jstring jstr)
{
char* rtn = NULL;
jclass clsstring =
env->FindClass("java/lang/String");
jstring strencode =
env->NewStringUTF("utf-8");
jmethodID mid =
env->GetMethodID(clsstring, "getBytes",
"(Ljava/lang/String;)[B");
jbyteArray barr=
(jbyteArray)env->CallObjectMethod(jstr, mid,
strencode);
jsize alen =
env->GetArrayLength(barr);
jbyte* ba =
env->GetByteArrayElements(barr,JNI_FALSE);
if(alen > 0){
rtn = (char*)malloc(alen +
1);
memcpy(rtn, ba, alen);
rtn[alen] = 0;
}
env->ReleaseByteArrayElements(barr,
ba, 0);
return rtn;
}
using namespace std;
int main()
{
JavaVMOption options[2];
JNIEnv *env;
JavaVM *jvm;
JavaVMInitArgs vm_args;
long status;
jclass cls;
jmethodID mid;
jint square;
jboolean jnot;
jobject jobj;
options[0].optionString =
"-Djava.compiler=NONE";
options[1].optionString =
"-Djava.class.path=.";
//options[2].optionString = "-verbose:jni";
//用于跟蹤運(yùn)行時(shí)的信息
vm_args.version = JNI_VERSION_1_4; //
JDK版本號(hào)
vm_args.nOptions = 2;
vm_args.options = options;
vm_args.ignoreUnrecognized = JNI_TRUE;
status =
JNI_CreateJavaVM(&jvm,
(void**)&env, &vm_args);
if(status != JNI_ERR){
printf("create java jvm
success\n");
cls =
env->FindClass("com/test/MyTest");
// 在這里查找ava類
if(cls !=0){
printf("find
java class success\n");
// 構(gòu)造函數(shù)
mid =
env->GetMethodID(cls,"<init>","()V");
if(mid
!=0){
jobj=env->NewObject(cls,mid);
std::cout
<< "init ok"
<< std::endl;
}
//
調(diào)用add函數(shù)
mid =
env->GetStaticMethodID( cls, "add", "(II)I");
if(mid
!=0){
square
= env->CallStaticIntMethod( cls, mid, 5,5);
std::cout
<< square
<< std::endl;
}
//
調(diào)用judge函數(shù)
mid =
env->GetMethodID( cls, "judge","(Z)Z");
if(mid
!=0){
jnot
= env->CallBooleanMethod(jobj, mid, 1);
if(!jnot)
std::cout << "Boolean ok"
<< std::endl;
}
}
else{
fprintf(stderr,
"FindClass failed\n");
}
jvm->DestroyJavaVM();
fprintf(stdout, "Java VM
destory.\n");
return 0;
}
else{
printf("create java jvm
fail\n");
return -1;
}
}
編譯該C++程序(前提:Java環(huán)境已設(shè)置好,即JAVA_HOME、PATH、JRE_HOME、LD_LIBRARY_PATH)
[root@localhost jni]# g++ -o testjava testjava.cpp
-I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux
-L${JRE_HOME}/lib/i386/client -ljvm
編譯好后可以用ldd testjava查看其使用的鏈接庫(kù)的正確性。
運(yùn)行:
[root@localhost jni]# ./testjava
create java jvm success
find java class success
init ok
10
Boolean ok
Java VM destory.
JRE_HOME和LD_LIBRARY_PATH要設(shè)置好,編譯C++程序時(shí)要使用JRE_HOME下面的libjvm.so動(dòng)態(tài)庫(kù)(一開始我使用網(wǎng)上說的使用JAVA_HOME目錄下的libjvm.so,結(jié)果出現(xiàn)下面錯(cuò)誤
# An unexpected error has been detected by HotSpot Virtual
Machine:
#
# SIGSEGV (0xb) at pc=0xb6d3dbe3, pid=14454,
tid=2773482416
#
# Java VM: Java HotSpot(TM) Server VM (1.5.0_11-b03 mixed mode)
。。。。。
------The End-----
#####################################################################################
|