首先要强调的是,native方法不但可以传递Java的基本类型做参数,还可以传递更复杂的类型,比如String,数组,甚至自定义的类。这一切都可以在jni.h中找到答案。
1.Java基本类型的传递
用过Java的人都知道,Java中的基本类型包括boolean,byte,char,short,int,long,float,double这样几种,如果你用这几种类型做native方法的参数,当你通过javah-jni生成.h文件的时候,只要看一下生成的.h文件,就会一清二楚,这些类型分别对应的类型是jboolean,jbyte,jchar,jshort,jint,jlong,jfloat,jdouble。这几种类型几乎都可以当成对应的C++类型来用,所以没什么好说的。
2.String参数的传递
Java的String和C++的string是不能对等起来的,所以处理起来比较麻烦。先看一个例子,
classPrompt{
//nativemethodthatprintsapromptandreadsaline
privatenativeStringgetLine(Stringprompt);
publicstaticvoidmain(Stringargs[]){
Promptp=newPrompt();
Stringinput=p.getLine("Typealine:");
System.out.println("Usertyped:"+input);
}
static{
System.loadLibrary("Prompt");
}
}
在这个例子中,我们要实现一个native方法
StringgetLine(Stringprompt);
读入一个String参数,返回一个String值。
通过执行javah-jni得到的头文件是这样的
#include
#ifndef_Included_Prompt
#define_Included_Prompt
#ifdef__cplusplus
extern"C"{
#endif
JNIEXPORTjstringJNICALLJava_Prompt_getLine(JNIEnv*env,jobjectthis,jstringprompt);
#ifdef__cplusplus
}
#endif
#endif
jstring是JNI中对应于String的类型,但是和基本类型不同的是,jstring不能直接当作C++的string用。如果你用
cout< 编译器肯定会扔给你一个错误信息的。
其实要处理jstring有很多种方式,这里只讲一种我认为最简单的方式,看下面这个例子,
#include"Prompt.h"
#include
JNIEXPORTjstringJNICALLJava_Prompt_getLine(JNIEnv*env,jobjectobj,jstringprompt)
{
constchar*str;
str=env->GetStringUTFChars(prompt,false);
if(str==NULL){
returnNULL;/*OutOfMemoryErroralreadythrown*/
}
std::cout<
env->ReleaseStringUTFChars(prompt,str);
char*tmpstr="returnstringsucceeded";
jstringrtstr=env->NewStringUTF(tmpstr);
returnrtstr;
}
在上面的例子中,作为参数的prompt不能直接被C++程序使用,先做了如下转换
str=env->GetStringUTFChars(prompt,false);
将jstring类型变成一个char*类型。
返回的时候,要生成一个jstring类型的对象,也必须通过如下命令,
jstringrtstr=env->NewStringUTF(tmpstr);
这里用到的GetStringUTFChars和NewStringUTF都是JNI提供的处理String类型的函数,还有其他的函数这里就不一一列举了。
3.数组类型的传递
和String一样,JNI为Java基本类型的数组提供了j*Array类型,比如int[]对应的就是jintArray。来看一个传递int数组的例子,Java程序就不写了,
JNIEXPORTjintJNICALLJava_IntArray_sumArray(JNIEnv*env,jobjectobj,jintArrayarr)
{
jint*carr;
carr=env->GetIntArrayElements(arr,false);
if(carr==NULL){
return0;/*exceptionoccurred*/
}
jintsum=0;
for(inti=0;i<10;i++){
sum+=carr;
}
env->ReleaseIntArrayElements(arr,carr,0);
returnsum;
}
这个例子中的GetIntArrayElements和ReleaseIntArrayElements函数就是JNI提供用于处理int数组的函数。如果试图用arr的方式去访问jintArray类型,毫无疑问会出错。JNI还提供了另一对函数GetIntArrayRegion和ReleaseIntArrayRegion访问int数组,就不介绍了,对于其他基本类型的数组,方法类似。
4.二维数组和String数组
在JNI中,二维数组和String数组都被视为object数组,因为数组和String被视为object。仍然用一个例子来说明,这次是一个二维int数组,作为返回值。
JNIEXPORTjobjectArrayJNICALLJava_ObjectArrayTest_initInt2DArray(JNIEnv*env,jclasscls,intsize)
{
jobjectArrayresult;
jclassintArrCls=env->FindClass("[I");
result=env->NewObjectArray(size,intArrCls,NULL);
for(inti=0;i
jinttmp[256];/*makesureitislargeenough!*/
jintArrayiarr=env->NewIntArray(size);
for(intj=0;j
tmp[j]=i+j;
}
env->SetIntArrayRegion(iarr,0,size,tmp);
env->SetObjectArrayElement(result,i,iarr);
env->DeleteLocalRef(iarr);
}
returnresult;
}
上面代码中的第三行,
jobjectArrayresult;
因为要返回值,所以需要新建一个jobjectArray对象。
jclassintArrCls=env->FindClass("[I");
是创建一个jclass的引用,因为result的元素是一维int数组的引用,所以intArrCls必须是一维int数组的引用,这一点是如何保证的呢?注意FindClass的参数"[I",JNI就是通过它来确定引用的类型的,I表示是int类型,[标识是数组。对于其他的类型,都有相应的表示方法,
Zboolean
Bbyte
Cchar
Sshort
Iint
Jlong
Ffloat
Ddouble
String是通过“Ljava/lang/String;”表示的,那相应的,String数组就应该是“[Ljava/lang/String;”。
还是回到代码,
result=env->NewObjectArray(size,intArrCls,NULL);
的作用是为result分配空间。
jintArrayiarr=env->NewIntArray(size);
是为一维int数组iarr分配空间。
env->SetIntArrayRegion(iarr,0,size,tmp);
是为iarr赋值。
env->SetObjectArrayElement(result,i,iarr);
是为result的第i个元素赋值。
通过上面这些步骤,我们就创建了一个二维int数组,并赋值完毕,这样就可以做为参数返回了。
如果了解了上面介绍的这些内容,基本上大部分的任务都可以对付了。虽然在操作数组类型,尤其是二维数组和String数组的时候,比起在单独的语言中编程要麻烦,但既然我们享受了跨语言编程的好处,必然要付出一定的代价。
有一点要补充的是,本文所用到的函数调用方式都是针对C++的,如果要在C中使用,所有的env->都要被替换成(*env)->,而且后面的函数中需要增加一个参数env,具体请看一下jni.h的代码。另外还有些省略的内容,可以参考JNI的文档:JavaNativeInterface6.0Specification,在JDK的文档里就可以找到。如果要进行更深入的JNI编程,需要仔细阅读这个文档。接下来的高级篇,也会讨论更深入的话题。
关于JNI编程更深入的话题,包括:在native方法中访问Java类的域和方法,将Java中自定义的类作为参数和返回值传递等等。了解这些内容,将会对JNI编程有更深入的理解,写出的程序也更清晰,易用性更好。
1.在一般的Java类中定义native方法
在前两篇的例子中,都是将native方法放在main方法的Java类中,实际上,完全可以在任何类中定义native方法。这样,对于外部来说,这个类和其他的Java类没有任何区别。
2.访问Java类的域和方法
native方法虽然是native的,但毕竟是方法,那么就应该同其他方法一样,能够访问类的私有域和方法。实际上,JNI的确可以做到这一点,我们通过几个例子来说明,
publicclassClassA{
Stringstr_="abcde";
intnumber_;
publicnativevoidnativeMethod();
privatevoidjavaMethod(){
System.out.println("calljavamethodsucceeded");
}
static{
System.loadLibrary("ClassA");
}
}
在这个例子中,我们在一个没有main方法的Java类中定义了native方法。我们将演示如何在nativeMethod()中访问域str_,number_和方法javaMethod(),nativeMethod()的C++实现如下,
JNIEXPORTvoidJNICALLJava_testclass_ClassCallDLL_nativeMethod(JNIEnv*env,jobjectobj){
//accessfield
jclasscls=env->GetObjectClass(obj);
jfieldIDfid=env->GetFieldID(cls,"str_","Ljava/lang/String;");
jstringjstr=(jstring)env->GetObjectField(obj,fid);
constchar*str=env->GetStringUTFChars(jstr,false);
if(std::string(str)=="abcde")
std::cout<<"accessfieldsucceeded"<
jinti=2468;
fid=env->GetFieldID(cls,"number_","I");
env->SetIntField(obj,fid,i);
//accessmethod
jmethodIDmid=env->GetMethodID(cls,"javaMethod","()V");
env->CallVoidMethod(obj,mid);
}
上面的代码中,通过如下两行代码获得str_的值,
jfieldIDfid=env->GetFieldID(cls,"str_","Ljava/lang/String;");
jstringjstr=(jstring)env->GetObjectField(obj,fid);
第一行代码获得str_的id,在GetFieldID函数的调用中需要指定str_的类型,第二行代码通过str_的id获得它的值,当然我们读到的是一个jstring类型,不能直接显示,需要转化为char*类型。
接下来我们看如何给Java类的域赋值,看下面两行代码,
fid=env->GetFieldID(cls,"number_","I");
env->SetIntField(obj,fid,i);
第一行代码同前面一样,获得number_的id,第二行我们通过SetIntField函数将i的值赋给number_,其他类似的函数可以参考JDK的文档。
访问javaMethod()的过程同访问域类似,
jmethodIDmid=env->GetMethodID(cls,"javaMethod","()V");
env->CallVoidMethod(obj,mid);
需要强调的是,在GetMethodID中,我们需要指定javaMethod方法的类型,域的类型很容易理解,方法的类型如何定义呢,在上面的例子中,我们用的是()V,V表示返回值为空,()表示参数为空。如果是更复杂的函数类型如何表示?看一个例子,
longf(intn,Strings,int[]arr);
这个函数的类型符号是(ILjava/lang/String;[I)J,I表示int类型,Ljava/lang/String;表示String类型,[I表示int数组,J表示long。这些都可以在文档中查到。
3.在native方法中使用用户定义的类
JNI不仅能使用Java的基础类型,还能使用用户定义的类,这样灵活性就大多了。大体上使用自定义的类和使用Java的基础类(比如String)没有太大的区别,关键的一点是,如果要使用自定义类,首先要能访问类的构造函数,看下面这一段代码,我们在native方法中使用了自定义的Java类ClassB,
jclasscls=env->FindClass("Ltestclass/ClassB;");
jmethodIDid=env->GetMethodID(cls,"","(D)V");
jdoubledd=0.033;
jvalueargs[1];
args[0].d=dd;
jobjectobj=env->NewObjectA(cls,id,args);
首先要创建一个自定义类的引用,通过FindClass函数来完成,参数同前面介绍的创建String对象的引用类似,只不过类名称变成自定义类的名称。然后通过GetMethodID函数获得这个类的构造函数,注意这里方法的名称是"",它表示这是一个构造函数。
jobjectobj=env->NewObjectA(cls,id,args);
生成了一个ClassB的对象,args是ClassB的构造函数的参数,它是一个jvalue*类型。
通过以上介绍的三部分内容,native方法已经看起来完全像Java自己的方法了,至少主要功能上齐备了,只是实现上稍麻烦。而了解了这些,JNI编程的水平也更上一层楼。下面要讨论的话题也是一个重要内容,至少如果没有它,我们的程序只能停留在演示阶段,不具有实用价值。
4.异常处理
在C++和Java的编程中,异常处理都是一个重要的内容。但是在JNI中,麻烦就来了,native方法是通过C++实现的,如果在native方法中发生了异常,如何传导到Java呢?
JNI提供了实现这种功能的机制。我们可以通过下面这段代码抛出一个Java可以接收的异常,
jclasserrCls;
env->ExceptionDescribe();
env->ExceptionClear();
errCls=env->FindClass("java/lang/IllegalArgumentException");
env->ThrowNew(errCls,"thrownfromC++code");
如果要抛出其他类型的异常,替换掉FindClass的参数即可。这样,在Java中就可以接收到native方法中抛出的异常。
1.Java基本类型的传递
用过Java的人都知道,Java中的基本类型包括boolean,byte,char,short,int,long,float,double这样几种,如果你用这几种类型做native方法的参数,当你通过javah-jni生成.h文件的时候,只要看一下生成的.h文件,就会一清二楚,这些类型分别对应的类型是jboolean,jbyte,jchar,jshort,jint,jlong,jfloat,jdouble。这几种类型几乎都可以当成对应的C++类型来用,所以没什么好说的。
2.String参数的传递
Java的String和C++的string是不能对等起来的,所以处理起来比较麻烦。先看一个例子,
classPrompt{
//nativemethodthatprintsapromptandreadsaline
privatenativeStringgetLine(Stringprompt);
publicstaticvoidmain(Stringargs[]){
Promptp=newPrompt();
Stringinput=p.getLine("Typealine:");
System.out.println("Usertyped:"+input);
}
static{
System.loadLibrary("Prompt");
}
}
在这个例子中,我们要实现一个native方法
StringgetLine(Stringprompt);
读入一个String参数,返回一个String值。
通过执行javah-jni得到的头文件是这样的
#include
#ifndef_Included_Prompt
#define_Included_Prompt
#ifdef__cplusplus
extern"C"{
#endif
JNIEXPORTjstringJNICALLJava_Prompt_getLine(JNIEnv*env,jobjectthis,jstringprompt);
#ifdef__cplusplus
}
#endif
#endif
jstring是JNI中对应于String的类型,但是和基本类型不同的是,jstring不能直接当作C++的string用。如果你用
cout< 编译器肯定会扔给你一个错误信息的。
其实要处理jstring有很多种方式,这里只讲一种我认为最简单的方式,看下面这个例子,
#include"Prompt.h"
#include
JNIEXPORTjstringJNICALLJava_Prompt_getLine(JNIEnv*env,jobjectobj,jstringprompt)
{
constchar*str;
str=env->GetStringUTFChars(prompt,false);
if(str==NULL){
returnNULL;/*OutOfMemoryErroralreadythrown*/
}
std::cout<
env->ReleaseStringUTFChars(prompt,str);
char*tmpstr="returnstringsucceeded";
jstringrtstr=env->NewStringUTF(tmpstr);
returnrtstr;
}
在上面的例子中,作为参数的prompt不能直接被C++程序使用,先做了如下转换
str=env->GetStringUTFChars(prompt,false);
将jstring类型变成一个char*类型。
返回的时候,要生成一个jstring类型的对象,也必须通过如下命令,
jstringrtstr=env->NewStringUTF(tmpstr);
这里用到的GetStringUTFChars和NewStringUTF都是JNI提供的处理String类型的函数,还有其他的函数这里就不一一列举了。
3.数组类型的传递
和String一样,JNI为Java基本类型的数组提供了j*Array类型,比如int[]对应的就是jintArray。来看一个传递int数组的例子,Java程序就不写了,
JNIEXPORTjintJNICALLJava_IntArray_sumArray(JNIEnv*env,jobjectobj,jintArrayarr)
{
jint*carr;
carr=env->GetIntArrayElements(arr,false);
if(carr==NULL){
return0;/*exceptionoccurred*/
}
jintsum=0;
for(inti=0;i<10;i++){
sum+=carr;
}
env->ReleaseIntArrayElements(arr,carr,0);
returnsum;
}
这个例子中的GetIntArrayElements和ReleaseIntArrayElements函数就是JNI提供用于处理int数组的函数。如果试图用arr的方式去访问jintArray类型,毫无疑问会出错。JNI还提供了另一对函数GetIntArrayRegion和ReleaseIntArrayRegion访问int数组,就不介绍了,对于其他基本类型的数组,方法类似。
4.二维数组和String数组
在JNI中,二维数组和String数组都被视为object数组,因为数组和String被视为object。仍然用一个例子来说明,这次是一个二维int数组,作为返回值。
JNIEXPORTjobjectArrayJNICALLJava_ObjectArrayTest_initInt2DArray(JNIEnv*env,jclasscls,intsize)
{
jobjectArrayresult;
jclassintArrCls=env->FindClass("[I");
result=env->NewObjectArray(size,intArrCls,NULL);
for(inti=0;i
jinttmp[256];/*makesureitislargeenough!*/
jintArrayiarr=env->NewIntArray(size);
for(intj=0;j
tmp[j]=i+j;
}
env->SetIntArrayRegion(iarr,0,size,tmp);
env->SetObjectArrayElement(result,i,iarr);
env->DeleteLocalRef(iarr);
}
returnresult;
}
上面代码中的第三行,
jobjectArrayresult;
因为要返回值,所以需要新建一个jobjectArray对象。
jclassintArrCls=env->FindClass("[I");
是创建一个jclass的引用,因为result的元素是一维int数组的引用,所以intArrCls必须是一维int数组的引用,这一点是如何保证的呢?注意FindClass的参数"[I",JNI就是通过它来确定引用的类型的,I表示是int类型,[标识是数组。对于其他的类型,都有相应的表示方法,
Zboolean
Bbyte
Cchar
Sshort
Iint
Jlong
Ffloat
Ddouble
String是通过“Ljava/lang/String;”表示的,那相应的,String数组就应该是“[Ljava/lang/String;”。
还是回到代码,
result=env->NewObjectArray(size,intArrCls,NULL);
的作用是为result分配空间。
jintArrayiarr=env->NewIntArray(size);
是为一维int数组iarr分配空间。
env->SetIntArrayRegion(iarr,0,size,tmp);
是为iarr赋值。
env->SetObjectArrayElement(result,i,iarr);
是为result的第i个元素赋值。
通过上面这些步骤,我们就创建了一个二维int数组,并赋值完毕,这样就可以做为参数返回了。
如果了解了上面介绍的这些内容,基本上大部分的任务都可以对付了。虽然在操作数组类型,尤其是二维数组和String数组的时候,比起在单独的语言中编程要麻烦,但既然我们享受了跨语言编程的好处,必然要付出一定的代价。
有一点要补充的是,本文所用到的函数调用方式都是针对C++的,如果要在C中使用,所有的env->都要被替换成(*env)->,而且后面的函数中需要增加一个参数env,具体请看一下jni.h的代码。另外还有些省略的内容,可以参考JNI的文档:JavaNativeInterface6.0Specification,在JDK的文档里就可以找到。如果要进行更深入的JNI编程,需要仔细阅读这个文档。接下来的高级篇,也会讨论更深入的话题。
关于JNI编程更深入的话题,包括:在native方法中访问Java类的域和方法,将Java中自定义的类作为参数和返回值传递等等。了解这些内容,将会对JNI编程有更深入的理解,写出的程序也更清晰,易用性更好。
1.在一般的Java类中定义native方法
在前两篇的例子中,都是将native方法放在main方法的Java类中,实际上,完全可以在任何类中定义native方法。这样,对于外部来说,这个类和其他的Java类没有任何区别。
2.访问Java类的域和方法
native方法虽然是native的,但毕竟是方法,那么就应该同其他方法一样,能够访问类的私有域和方法。实际上,JNI的确可以做到这一点,我们通过几个例子来说明,
publicclassClassA{
Stringstr_="abcde";
intnumber_;
publicnativevoidnativeMethod();
privatevoidjavaMethod(){
System.out.println("calljavamethodsucceeded");
}
static{
System.loadLibrary("ClassA");
}
}
在这个例子中,我们在一个没有main方法的Java类中定义了native方法。我们将演示如何在nativeMethod()中访问域str_,number_和方法javaMethod(),nativeMethod()的C++实现如下,
JNIEXPORTvoidJNICALLJava_testclass_ClassCallDLL_nativeMethod(JNIEnv*env,jobjectobj){
//accessfield
jclasscls=env->GetObjectClass(obj);
jfieldIDfid=env->GetFieldID(cls,"str_","Ljava/lang/String;");
jstringjstr=(jstring)env->GetObjectField(obj,fid);
constchar*str=env->GetStringUTFChars(jstr,false);
if(std::string(str)=="abcde")
std::cout<<"accessfieldsucceeded"<
jinti=2468;
fid=env->GetFieldID(cls,"number_","I");
env->SetIntField(obj,fid,i);
//accessmethod
jmethodIDmid=env->GetMethodID(cls,"javaMethod","()V");
env->CallVoidMethod(obj,mid);
}
上面的代码中,通过如下两行代码获得str_的值,
jfieldIDfid=env->GetFieldID(cls,"str_","Ljava/lang/String;");
jstringjstr=(jstring)env->GetObjectField(obj,fid);
第一行代码获得str_的id,在GetFieldID函数的调用中需要指定str_的类型,第二行代码通过str_的id获得它的值,当然我们读到的是一个jstring类型,不能直接显示,需要转化为char*类型。
接下来我们看如何给Java类的域赋值,看下面两行代码,
fid=env->GetFieldID(cls,"number_","I");
env->SetIntField(obj,fid,i);
第一行代码同前面一样,获得number_的id,第二行我们通过SetIntField函数将i的值赋给number_,其他类似的函数可以参考JDK的文档。
访问javaMethod()的过程同访问域类似,
jmethodIDmid=env->GetMethodID(cls,"javaMethod","()V");
env->CallVoidMethod(obj,mid);
需要强调的是,在GetMethodID中,我们需要指定javaMethod方法的类型,域的类型很容易理解,方法的类型如何定义呢,在上面的例子中,我们用的是()V,V表示返回值为空,()表示参数为空。如果是更复杂的函数类型如何表示?看一个例子,
longf(intn,Strings,int[]arr);
这个函数的类型符号是(ILjava/lang/String;[I)J,I表示int类型,Ljava/lang/String;表示String类型,[I表示int数组,J表示long。这些都可以在文档中查到。
3.在native方法中使用用户定义的类
JNI不仅能使用Java的基础类型,还能使用用户定义的类,这样灵活性就大多了。大体上使用自定义的类和使用Java的基础类(比如String)没有太大的区别,关键的一点是,如果要使用自定义类,首先要能访问类的构造函数,看下面这一段代码,我们在native方法中使用了自定义的Java类ClassB,
jclasscls=env->FindClass("Ltestclass/ClassB;");
jmethodIDid=env->GetMethodID(cls,"","(D)V");
jdoubledd=0.033;
jvalueargs[1];
args[0].d=dd;
jobjectobj=env->NewObjectA(cls,id,args);
首先要创建一个自定义类的引用,通过FindClass函数来完成,参数同前面介绍的创建String对象的引用类似,只不过类名称变成自定义类的名称。然后通过GetMethodID函数获得这个类的构造函数,注意这里方法的名称是"",它表示这是一个构造函数。
jobjectobj=env->NewObjectA(cls,id,args);
生成了一个ClassB的对象,args是ClassB的构造函数的参数,它是一个jvalue*类型。
通过以上介绍的三部分内容,native方法已经看起来完全像Java自己的方法了,至少主要功能上齐备了,只是实现上稍麻烦。而了解了这些,JNI编程的水平也更上一层楼。下面要讨论的话题也是一个重要内容,至少如果没有它,我们的程序只能停留在演示阶段,不具有实用价值。
4.异常处理
在C++和Java的编程中,异常处理都是一个重要的内容。但是在JNI中,麻烦就来了,native方法是通过C++实现的,如果在native方法中发生了异常,如何传导到Java呢?
JNI提供了实现这种功能的机制。我们可以通过下面这段代码抛出一个Java可以接收的异常,
jclasserrCls;
env->ExceptionDescribe();
env->ExceptionClear();
errCls=env->FindClass("java/lang/IllegalArgumentException");
env->ThrowNew(errCls,"thrownfromC++code");
如果要抛出其他类型的异常,替换掉FindClass的参数即可。这样,在Java中就可以接收到native方法中抛出的异常。
相关推荐
本文将深入解析Android系统中的JNI编程,包括其基本概念、开发步骤、注意事项以及实际应用。 1. JNI基本概念 JNI是Java平台提供的一种接口,它定义了一套标准的函数调用规范,使得Java代码可以调用C/C++编写的功能...
### Android深入研究JNI详解 #### 一、JNI与Android中的.so库载入 在深入了解JNI(Java Native Interface)之前,我们需要明确一点:Android应用程序的主要部分是基于Java开发的,并且这些Java类会被编译成Dex格式...
### Android开发书籍 - JNI详解 #### 一、前言与翻译初衷 本书旨在记录JNI(Java Native Interface)编程的经验,并以此为基础,探索更深层次的虚拟机相关知识。作者原本计划编写一本更为深入的虚拟机书籍,但意识...
- **JNI编程入门**:本书的第2章通过简单的示例介绍了JNI编程的基础知识,帮助初学者快速上手。 - **JNI的高级特性**:从第3章到第10章,本书深入探讨了JNI的各种特性和功能,并提供了相应的示例代码,这对于希望...
Android 开发书籍 - JNI 详解导航版 本书籍主要介绍了 Java Native Interface(JNI)的内容,并探讨了在 Android 开发中 JNI 的应用。JNI 是 SUN 定义的一套标准接口,允许 Java 代码调用本地代码,本地代码也可以...
作者在前言中表达了对JNI编程的热爱及对未来深入研究虚拟机技术的渴望,同时也透露了对现有水平的谦逊认知,认为达到精通虚拟机编写的目标尚需时日。尽管如此,作者仍希望通过本书的编写,促进对虚拟机、编译器、...
Android 深入研究 JNI 详解 JNI(Java Native Interface)是 Java 平台的一部分,允许 Java 代码和其他语言编写的代码进行交互。JNI 是本地编程接口,使得在 Java 虚拟机(VM)内部运行的 Java 代码能够与用其他...
### JNI编程指南书知识点概述 #### 一、JNI的基本概念及应用场景 JNI(Java Native Interface)作为Java平台的重要组成部分,其核心价值在于能够使Java程序有效地利用现有的C/C++代码,进而实现性能优化或功能扩展...
2. **JNI**:JNI是Java平台的标准编程接口,允许Java代码和其他语言写的代码进行交互。在Android中,JNI通常用来调用C/C++的原生库,提高性能或利用现有的C/C++代码库。在这个案例中,JNI被用来实现RSA和3DES的加密...
通过这些项目的实践,可以深入了解Android开发中的关键技术点,如多线程、自定义UI组件、网络通信、JNI编程以及串口通信等。这些技能对于从事Android开发的专业人士来说是非常宝贵的。此外,项目中提到的一些具体...
7. **错误处理与调试**:JNI编程中常见的错误和调试方法,如JNI异常处理、日志输出等。 8. **性能优化**:使用JNI进行性能优化的策略,如减少Java-C++间的函数调用、缓存JNIEnv指针等。 9. **实例分析**:可能包含...
### Android JNI详解 #### 一、概述 Java Native Interface (JNI) 是一种强大的技术,它允许Java代码调用本地代码(如C/C++),从而在Android应用开发中发挥重要作用。JNI不仅增强了应用性能,还提供了丰富的功能...
《Android JNI之OpenSSL详解》 在Android应用开发中,JNI(Java Native Interface)是一种连接Java和本地代码的桥梁,允许开发者使用C/C++代码来实现高性能或特定平台的功能。OpenSSL是一个强大的安全套接层密码库...
《Android OLED JNI 应用开发详解》 在移动设备领域,Android系统占据着主导地位,而随着OLED屏幕的普及,开发者们对于如何在Android平台上高效地利用这种高对比度、低功耗的显示屏产生了浓厚的兴趣。"我的android ...
- 阅读`Java本地接口(JNI)编程指南和规范`及`JNI编程指南`,了解实际开发中的细节和技巧。 - 深入研究`jni详解.pdf`,掌握更复杂的JNI使用方式和最佳实践。 - 实践项目,按照`android之JNI开发步骤总结.doc`的...
《Android 底层接口与驱动开发技术详解》是一本深入探讨Android系统底层接口和驱动开发的专业书籍,适合对Android系统有深入研究需求的开发者阅读。本书涵盖了从硬件抽象层(HAL)到用户空间驱动,再到应用程序接口...
### Android JNI编程详解 #### 一、概述 JNI(Java Native Interface)是Java平台标准的一部分,它允许Java代码和其他语言写的代码进行交互。在Android开发中,JNI常用于调用C/C++编写的本地库代码,以此来实现...
1. **配置开发环境**:确保你的Android Studio已经安装了NDK,这是开发JNI程序所必需的。如果没有,需要先进行安装。同时,熟悉JNI的使用也是必要的。 2. **创建C++项目**:在Android Studio中,创建一个新的支持...
Android进阶初级:组件Widget/ 菜单Menu/ 布局Layout 详解 Xml解析(Pull/Dom/Sax)/JNI 解析SQL数据库原理, SQLit e /SharedPreferences/File详解 多媒体Audio/Video/Camera 详解 Android进阶高级:蓝牙/WIFI SMS/...