首页 > 代码库 > JNI笔记之 初体验

JNI笔记之 初体验

Java Native Interface提供了java与c语言写的代码之间互相调用的方式。在c语言方面jni.h中声明了许多的类型和方法,有很多java的数据类型和c语言类型的转换方法函数。

java里的int,String,byte[]等对应于C方面的jint,jstring,jbyteArray.int可以直接赋给jint型的变量。

Java的String和C++的string是不能对等起来的,所以jstring的操作较为繁琐,通常可转为c里面的char *,有两种方式,先上简单的:

JNIEXPORT jstring JNICALL Java_Prompt_getLine(JNIEnv*env,jobject obj,jstring prompt)
  
  {
  
  const char*str;
  
  str=env->GetStringUTFChars(prompt,false);
  
  if(str==NULL){
  
  return NULL;/*Out Of Memory Error already thrown*/
   
  }
  
  std::cout<<str<<std::endl;
  
  env->ReleaseStringUTFChars(prompt,str);
  
  
  char*  tmpstr="return string succeeded";
  
  jstring rtstr=env->NewStringUTF(tmpstr);
  
  return rtstr;
  
  }
  
  在上面的例子中,作为参数的prompt不能直接被C++程序使用,先做了如下转换
  str=env->GetStringUTFChars(prompt,false);
  将jstring类型变成一个char*类型。
  
  返回的时候,要生成一个jstring类型的对象,也必须通过如下命令,
  jstringrtstr=env->NewStringUTF(tmpstr);
  
  这里用到的GetStringUTFChars和NewStringUTF都是JNI提供的处理String类型的函数,还有其他的函数这里就不一一列举了。

数组操作有:jint*carr;  
  carr=env->GetIntArrayElements(arr,false);  

  env->ReleaseIntArrayElements(arr,carr,0);

env->SetIntArrayRegion(jintarr,i,num,&tmp);//修改jintarr的从i位置开始的连续num个数为tmp里的值

注意上边的env在C++代码里是一个指向java虚拟机(JVM)的一个指针,(不知说的对不对,咱也不是专家,这也许在android里有些区别,android用了很多年的Dalvik虚拟机,每个app为一个子进程,通过NDK写的动态链接库执行时与java的代码执行在一个进程里),在c语言里面env是个二级指针,使用时形如:

(*env)->SetIntArrayRegion(env,jintarr,i,1,&tmp);参数里面需要传递env。

c中调用java中的函数的方式,类似反射机制:

char* classname="com/example/calculate";//类名,类中有个方法add(int a,int b)

jclass clazz;

clazz=(*env)->FindClass(env,classname);

if(clazz==0) //printf or LOGI("Can‘t find clazz");return;

jmethodID java_method=(*env)->GetMethodID(env,clazz,"add","(II)I");//方法名为add,签名为(II)I,可以用javap查看

if(java_method==0){...}//not find

jint result=(*env)->CallIntMethod(env,obj,java_method,5,8);//obj为jobject类型通常有主函数传递来的//结果返回5+8;

 再利用这种方式实现jstring与char*之间的互相转换:

//将const char类型转换成jstring类型
jstring CStr2Jstring( JNIEnv* env, const char* pat )
{
 //定义java String类 strClass
 jclass strClass = (env)->FindClass("Ljava/lang/String;");
 //获取java String类方法String(byte[],String)的构造器,用于将本地byte[]数组转换为一个新String
 jmethodID ctorID = (env)->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");
 //建立byte数组
 jbyteArray bytes = (env)->NewByteArray((jsize)strlen(pat));
 //将char* 转换为byte数组
 (env)->SetByteArrayRegion(bytes, 0, (jsize)strlen(pat), (jbyte*)pat);
 //设置String, 保存语言类型,用于byte数组转换至String时的参数
 jstring encoding = (env)->NewStringUTF("GB2312"); 
 //将byte数组转换为java String,并输出
 return (jstring)(env)->NewObject(strClass, ctorID, bytes, encoding);
}

char*   Jstring2CStr(JNIEnv*   env,   jstring   jstr)  
{  
 char*   rtn   =   NULL;  
 jclass   clsstring   =   env->FindClass("java/lang/String");   
 jstring   strencode   =   env->NewStringUTF("GB2312");  
 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);         //new   char[alen+1];  
  memcpy(rtn,ba,alen);  
  rtn[alen]=0;  
 }  
 env->ReleaseByteArrayElements(barr,ba,0);  
 return rtn;
}