1. 程式人生 > >Android C++和JAVA互相呼叫

Android C++和JAVA互相呼叫

 

摘要:

Java類生成c標頭檔案和庫檔案

2 對於c/c++程式,啟動時先啟動jvm,然後獲得對應的java類的物件和方法。然後正常使用。

最近正在做一個C/C++呼叫java的程式,這裡說的呼叫java不是使用方式 exec(/path/to/java,.....),而是呼叫一個class檔案中的一個特定的函式。

實踐後總結如下:

1. 安裝 jdk

2. 安裝gcc(Linux自帶有的就無需安裝了)

利用JNI(Java native interface),來實現動態建立java runtime environment.

第一,C/C++程式中包含標頭檔案"jni.h"

#include <jni.h> 一般在JAVA_HOME/include 目錄下。

呼叫jni.h中的方法建立runtime env 然後呼叫java 程式。

第二,編譯

g++ -o testjava testjava.cpp -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux -L${JRE_HOME}/lib/i386/client -ljvm

以上就是大致思路,現詳細說明過程如下:

#####################################################################################

一、安裝配置Java環境

我的linux是RedHat Enterprise linux 5, 核心版本2.6.18
在Linux系統中安裝Java比較簡單。可以訪問Java download網站或自由軟體庫等,選擇你所有安裝的

作業系統型別(Linux,Linux AMD64,Solaris等)。一旦你已經選擇下載檔案──要麼是自解壓縮執行檔案,要麼是自解壓縮的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

執行完後生成jdk1.5.0_06目錄,jdk被安裝在/usr/local/java/jdk1.5.0_06/。執行以下執行程式碼將得到一個

測試結果:

# cd jdk1.5.0_06/bin

[[email protected] 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,需要設定如下環境變數:

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的配置,若機器上沒有jre環境,則安裝jre,安裝方法類似安裝jdk

設定完後可以檢視變數的值

[[email protected] bin]# echo $JAVA_HOME

/usr/local/java/jdk1.5.0_06

[[email protected] 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

二、編寫簡單的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,將其放置於當前目錄的com/test目錄下,C++程式的JNI呼叫時會使用相關方法在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"; //用於跟蹤執行時的資訊

 vm_args.version = JNI_VERSION_1_4; // JDK版本號
 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");
   // 建構函式
   mid = env->GetMethodID(cls,"<init>","()V");
   if(mid !=0){
    jobj=env->NewObject(cls,mid);
    std::cout << "init ok" << std::endl;
   }
       
   // 呼叫add函式
   mid = env->GetStaticMethodID( cls, "add", "(II)I");
   if(mid !=0){
    square = env->CallStaticIntMethod( cls, mid, 5,5);
    std::cout << square << std::endl;
   }


   // 呼叫judge函式
   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環境已設定好,即JAVA_HOME、PATH、JRE_HOME、LD_LIBRARY_PATH)
[[email protected] jni]# g++ -o testjava testjava.cpp -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux -L${JRE_HOME}/lib/i386/client -ljvm

編譯好後可以用ldd testjava檢視其使用的連結庫的正確性。

執行:
[[email protected] jni]# ./testjava
create java jvm success
find java class success
init ok
10
Boolean ok
Java VM destory.


JRE_HOME和LD_LIBRARY_PATH要設定好,編譯C++程式時要使用JRE_HOME下面的libjvm.so動態庫(一開始我使用網上說的使用JAVA_HOME目錄下的libjvm.so,結果出現下面錯誤

# 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)

。。。。。