`
苗振忠
  • 浏览: 59672 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

JNI 实战全面解析

 
阅读更多

这篇讲的很详细,地址:点击打开链接

另外JNI视频讲解,留着看:点击打开链接

项目决定移植一款C++开源项目到Android平台,开始对JNI深入研究。

JNI是什么?

JNI(JavaNative Interface)意为JAVA本地调用,它允许Java代码和其他语言写的代码进行交互,简单的说,一种在Java虚拟机控制下执行代码的标准机制。

NDK是什么?

Android NDK(Native Development Kit )是一套工具集合,允许你用像C/C++语言那样实现应用程序的一部分。

为什么要用NDK?

1、安全性,java是半解释型语言,很容易被反汇编后拿到源代码文件,我们可以在重要的交互功能使用C语言代替。
2、效率,C语言比起java来说效率要高出很多。

JNI和NDK的区别?

从工具上说,NDK其实多了一个把.so和.apk打包的工具,而JNI开发并没有打包,只是把.so文件放到文件系统的特定位置。
从编译库说NDK开发C/C++只能能使用NDK自带的有限的头文件,而使用JNI则可以使用文件系统中带的头文件。
从编写方式说,它们一样。

详解
1、JNI 元素

1、JNI组织结构


JNI函数表的组成就像C++的虚函数表,虚拟机可以运行多张函数表。
JNI接口指针仅在当前线程中起作用,指针不能从一个线程进入另一个线程,但可以在不同的线程中调用本地方法。


2、原始数据

Jobject 对象 引用类型


Java类型 本地类型(JNI) 描述
boolean(布尔型) jboolean 无符号8个比特
byte(字节型) jbyte 有符号8个比特
char(字符型) jchar 无符号16个比特
short(短整型) jshort 有符号16个比特
int(整型) jint 有符号32个比特
long(长整型) jlong 有符号64个比特
float(浮点型) jfloat 32个比特
double(双精度浮点型) jdouble 64个比特
void(空型) void N/A

函数操作

函数 Java 数组类型 本地类型 说明
GetBooleanArrayElements jbooleanArray jboolean ReleaseBooleanArrayElements 释放
GetByteArrayElements jbyteArray jbyte ReleaseByteArrayElements 释放
GetCharArrayElements jcharArray jchar ReleaseShortArrayElements 释放
GetShortArrayElements jshortArray jshort ReleaseBooleanArrayElements 释放
GetIntArrayElements jintArray jint ReleaseIntArrayElements 释放
GetLongArrayElements jlongArray jlong ReleaseLongArrayElements 释放
GetFloatArrayElements jfloatArray jfloat ReleaseFloatArrayElements 释放
GetDoubleArrayElements jdoubleArray jdouble ReleaseDoubleArrayElements 释放
GetObjectArrayElement 自定义对象 object
SetObjectArrayElement 自定义对象 object
GetArrayLength 获取数组大小
New<Type>Array 创建一个指定长度的原始数据类型的数组
GetPrimitiveArrayCritical 得到指向原始数据类型内容的指针,该方法可能使垃圾回收不能执行,该方法可能返回数组的拷贝,因此必须释放此资源。
ReleasePrimitiveArrayCritical 释放指向原始数据类型内容的指针,该方法可能使垃圾回收不能执行,该方法可能返回数组的拷贝,因此必须释放此资源。
NewStringUTF jstring类型的方法转换
GetStringUTFChars jstring类型的方法转换
DefineClass 从原始类数据的缓冲区中加载类
FindClass 该函数用于加载本地定义的类。它将搜索由CLASSPATH 环境变量为具有指定名称的类所指定的目录和 zip文件
GetObjectClass 通过对象获取这个类。该函数比较简单,唯一注意的是对象不能为NULL,否则获取的class肯定返回也为NULL
GetSuperclass 获取父类或者说超类 。 如果 clazz 代表类class而非类 object,则该函数返回由 clazz 所指定的类的超类。 如果 clazz指定类 object 或代表某个接口,则该函数返回NULL
IsAssignableFrom 确定 clazz1 的对象是否可安全地强制转换为clazz2
Throw 抛出 java.lang.Throwable 对象
ThrowNew 利用指定类的消息(由 message 指定)构造异常对象并抛出该异常
ExceptionOccurred 确定是否某个异常正被抛出。在平台相关代码调用 ExceptionClear() 或 Java 代码处理该异常前,异常将始终保持抛出状态
ExceptionDescribe 将异常及堆栈的回溯输出到系统错误报告信道(例如 stderr)。该例程可便利调试操作
ExceptionClear 清除当前抛出的任何异常。如果当前无异常,则此例程不产生任何效果
FatalError 抛出致命错误并且不希望虚拟机进行修复。该函数无返回值
NewGlobalRef 创建 obj 参数所引用对象的新全局引用。obj 参数既可以是全局引用,也可以是局部引用。全局引用通过调用DeleteGlobalRef() 来显式撤消。
DeleteGlobalRef 删除 globalRef 所指向的全局引用
DeleteLocalRef 删除 localRef所指向的局部引用
AllocObject 分配新 Java 对象而不调用该对象的任何构造函数。返回该对象的引用。clazz 参数务必不要引用数组类。
getObjectClass 返回对象的类
IsSameObject 测试两个引用是否引用同一 Java 对象
NewString 利用 Unicode 字符数组构造新的 java.lang.String 对象
GetStringLength 返回 Java 字符串的长度(Unicode 字符数)
GetStringChars 返回指向字符串的 Unicode 字符数组的指针。该指针在调用 ReleaseStringchars() 前一直有效
ReleaseStringChars 通知虚拟机平台相关代码无需再访问 chars。参数chars 是一个指针,可通过 GetStringChars() 从 string 获得
NewStringUTF 利用 UTF-8 字符数组构造新 java.lang.String 对象
GetStringUTFLength 以字节为单位返回字符串的 UTF-8 长度
GetStringUTFChars 返回指向字符串的 UTF-8 字符数组的指针。该数组在被ReleaseStringUTFChars() 释放前将一直有效
ReleaseStringUTFChars 通知虚拟机平台相关代码无需再访问 utf。utf 参数是一个指针,可利用 GetStringUTFChars() 获得
NewObjectArray 构造新的数组,它将保存类 elementClass 中的对象。所有元素初始值均设为 initialElement
Set<PrimitiveType>ArrayRegion 将基本类型数组的某一区域从缓冲区中复制回来的一组函数
GetFieldID 返回类的实例(非静态)域的属性 ID。该域由其名称及签名指定。访问器函数的
Get<type>Field 及 Set<type>Field系列使用域 ID 检索对象域。GetFieldID() 不能用于获取数组的长度域。应使用GetArrayLength()。
Get<type>Field 该访问器例程系列返回对象的实例(非静态)域的值。要访问的域由通过调用GetFieldID()而得到的域 ID 指定。
Set<type>Field 该访问器例程系列设置对象的实例(非静态)属性的值。要访问的属性由通过调用
SetFieldID() 而得到的属性 ID指定。
GetStaticFieldID

GetStatic<type>Field

SetStatic<type>Field
同上,只不过是静态属性。
GetMethodID 返回类或接口实例(非静态)方法的方法 ID。方法可在某个 clazz 的超类中定义,也可从clazz 继承。该方法由其名称和签名决定。 GetMethodID() 可使未初始化的类初始化。要获得构造函数的方法 ID,应将<init> 作为方法名,同时将void (V) 作为返回类型。
CallVoidMethod
CallObjectMethod
CallBooleanMethod
CallByteMethod
CallCharMethod
CallShortMethod
CallIntMethod
CallLongMethod
CallFloatMethod
CallDoubleMethod
GetStaticMethodID 调用静态方法
Call<type>Method
RegisterNatives 向 clazz 参数指定的类注册本地方法。methods 参数将指定 JNINativeMethod 结构的数组,其中包含本地方法的名称、签名和函数指针。nMethods 参数将指定数组中的本地方法数。
UnregisterNatives 取消注册类的本地方法。类将返回到链接或注册了本地方法函数前的状态。该函数不应在常规平台相关代码中使用。相反,它可以为某些程序提供一种重新加载和重新链接本地库的途径。

域描述符

Java 语言
Z boolean
B byte
C char
S short
I int
J long
F float
D double

引用类型则为 L + 该类型类描述符 + 。

数组,其为 : [ + 其类型的域描述符 + 。

多维数组则是 n个[ +该类型的域描述符 , N代表的是几维数组。

  1. String类型的域描述符为Ljava/lang/String;
  2. [+其类型的域描述符+;
  3. int[]其描述符为[I
  4. float[]其描述符为[F
  5. String[]其描述符为[Ljava/lang/String;
  6. Object[]类型的域描述符为[Ljava/lang/Object;
  7. int[][]其描述符为[[I
  8. float[][]其描述符为[[F

将参数类型的域描述符按照申明顺序放入一对括号中后跟返回值类型的域描述符,规则如下: (参数的域描述符的叠加)返回类型描述符。对于,没有返回值的,用V(表示void型)表示。
举例如下:

  1. Java层方法JNI函数签名
  2. Stringtest()Ljava/lang/String;
  3. intf(inti,Objectobject)(ILjava/lang/Object;)I
  4. voidset(byte[]bytes)([B)V

JNIEnv与JavaVM

JNIEnv 概念 : 是一个线程相关的结构体, 该结构体代表了 Java 在本线程的运行环境 ;

JNIEnv 与 JavaVM : 注意区分这两个概念;
-- JavaVM : JavaVM 是 Java虚拟机在 JNI 层的代表, JNI 全局只有一个;
-- JNIEnv : JavaVM 在线程中的代表, 每个线程都有一个, JNI 中可能有很多个 JNIEnv;

JNIEnv 作用 :
-- 调用 Java 函数 : JNIEnv 代表 Java 运行环境, 可以使用 JNIEnv 调用 Java 中的代码;
-- 操作 Java 对象 : Java 对象传入 JNI 层就是 Jobject 对象, 需要使用 JNIEnv 来操作这个 Java 对象;


JNIEnv 体系结构

线程相关 : JNIEnv 是线程相关的, 即 在 每个线程中 都有一个 JNIEnv 指针, 每个JNIEnv 都是线程专有的, 其它线程不能使用本线程中的 JNIEnv, 线程 A 不能调用 线程 B 的 JNIEnv;

JNIEnv 不能跨线程 :
-- 当前线程有效 : JNIEnv 只在当前线程有效, JNIEnv 不能在 线程之间进行传递, 在同一个线程中, 多次调用 JNI层方法, 传入的 JNIEnv 是相同的;
-- 本地方法匹配多JNIEnv : 在 Java 层定义的本地方法, 可以在不同的线程调用, 因此 可以接受不同的 JNIEnv;

JNIEnv 结构 : 由上面的代码可以得出, JNIEnv 是一个指针, 指向一个线程相关的结构, 线程相关结构指向 JNI 函数指针 数组, 这个数组中存放了大量的 JNI 函数指针, 这些指针指向了具体的 JNI 函数;


注意:JNIEnv只在当前线程中有效。本地方法不能将JNIEnv从一个线程传递到另一个线程中。相同的 Java 线程中对本地方法多次调用时,传递给该本地方法的JNIEnv是相同的。但是,一个本地方法可被不同的 Java 线程所调用,因此可以接受不同的 JNIEnv。



UTF-8编码

JNI使用改进的UTF-8字符串来表示不同的字符类型。Java使用UTF-16编码。UTF-8编码主要使用于C语言,因为它的编码用\u000表示为0xc0,而不是通常的0×00。非空ASCII字符改进后的字符串编码中可以用一个字节表示。

错误

JNI不会检查NullPointerException、IllegalArgumentException这样的错误,原因是:导致性能下降。

在绝大多数C的库函数中,很难避免错误发生。
JNI允许用户使用Java异常处理。大部分JNI方法会返回错误代码但本身并不会报出异常。因此,很有必要在代码本身进行处理,将异常抛给Java。在JNI内部,首先会检查调用函数返回的错误代码,之后会调用ExpectOccurred()返回一个错误对象。
  1. jthrowableExceptionOccurred(JNIEnv*env);
  2. 例如:一些操作数组的JNI函数不会报错,因此可以调用ArrayIndexOutofBoundsException或ArrayStoreExpection方法报告异常。
3、JNI函数实战

1、*.so的入口函数

JNI_OnLoad()与JNI_OnUnload()
当Android的VM(Virtual Machine)执行到System.loadLibrary()函数时,首先会去执行C组件里的JNI_OnLoad()函数。它的用途有二:
(1)告诉VM此C组件使用那一个JNI版本。如果你的*.so档没有提供JNI_OnLoad()函数,VM会默认该*.so档是使用最老的JNI 1.1版本。由于新版的JNI做了许多扩充,如果需要使用JNI的新版功能,例如JNI 1.4的java.nio.ByteBuffer,就必须藉由JNI_OnLoad()函数来告知VM。
(2)由于VM执行到System.loadLibrary()函数时,就会立即先呼叫JNI_OnLoad(),所以C组件的开发者可以藉由JNI_OnLoad()来进行C组件内的初期值之设定(Initialization) 。


2、返回值

  1. jstringstr=env->newStringUTF("HelloJNI");//直接使用该JNI构造一个jstring对象返回
  2. returnstr;
  1. jobjectArrayret=0;
  2. jsizelen=5;
  3. jstringstr;
  4. stringvalue("hello");
  5. ret=(jobjectArray)(env->NewObjectArray(len,env->FindClass("java/lang/String"),0));
  6. for(inti=0;i<len;i++)
  7. {
  8. str=env->NewStringUTF(value..c_str());
  9. env->SetObjectArrayElement(ret,i,str);
  10. }
  11. returnret;返回数组
  1. jclassm_cls=env->FindClass("com/ldq/ScanResult");
  2. jmethodIDm_mid=env->GetMethodID(m_cls,"<init>","()V");
  3. jfieldIDm_fid_1=env->GetFieldID(m_cls,"ssid","Ljava/lang/String;");
  4. jfieldIDm_fid_2=env->GetFieldID(m_cls,"mac","Ljava/lang/String;");
  5. jfieldIDm_fid_3=env->GetFieldID(m_cls,"level","I");
  6. jobjectm_obj=env->NewObject(m_cls,m_mid);
  7. env->SetObjectField(m_obj,m_fid_1,env->NewStringUTF("AP1"));
  8. env->SetObjectField(m_obj,m_fid_2,env->NewStringUTF("00-11-22-33-44-55"));
  9. env->SetIntField(m_obj,m_fid_3,-50);
  10. returnm_obj;返回自定义对象

  1. jclasslist_cls=env->FindClass("Ljava/util/ArrayList;");//获得ArrayList类引用
  2. if(listcls==NULL)
  3. {
  4. cout<<"listclsisnull\n";
  5. }
  6. jmethodIDlist_costruct=env->GetMethodID(list_cls,"<init>","()V");//获得得构造函数Id
  7. jobjectlist_obj=env->NewObject(list_cls,list_costruct);//创建一个Arraylist集合对象
  8. //或得Arraylist类中的add()方法ID,其方法原型为:booleanadd(Objectobject);
  9. jmethodIDlist_add=env->GetMethodID(list_cls,"add","(Ljava/lang/Object;)Z");
  10. jclassstu_cls=env->FindClass("Lcom/feixun/jni/Student;");//获得Student类引用
  11. //获得该类型的构造函数函数名为<init>返回类型必须为void即V
  12. jmethodIDstu_costruct=env->GetMethodID(stu_cls,"<init>","(ILjava/lang/String;)V");
  13. for(inti=0;i<3;i++)
  14. {
  15. jstringstr=env->NewStringUTF("Native");
  16. //通过调用该对象的构造函数来new一个Student实例
  17. jobjectstu_obj=env->NewObject(stucls,stu_costruct,10,str);//构造一个对象
  18. env->CallBooleanMethod(list_obj,list_add,stu_obj);//执行Arraylist类实例的add方法,添加一个stu对象
  19. }
  20. returnlist_obj;返回对象集合

3、操作Java层的类

  1. //获得jfieldID以及该字段的初始值
  2. jfieldIDnameFieldId;
  3. jclasscls=env->GetObjectClass(obj);//获得Java层该对象实例的类引用,即HelloJNI类引用
  4. nameFieldId=env->GetFieldID(cls,"name","Ljava/lang/String;");//获得属性句柄
  5. if(nameFieldId==NULL)
  6. {
  7. cout<<"没有得到name的句柄Id\n;";
  8. }
  9. jstringjavaNameStr=(jstring)env->GetObjectField(obj,nameFieldId);//获得该属性的值
  10. constchar*c_javaName=env->GetStringUTFChars(javaNameStr,NULL);//转换为char*类型
  11. stringstr_name=c_javaName;
  12. cout<<"thenamefromjavais"<<str_name<<endl;//输出显示
  13. env->ReleaseStringUTFChars(javaNameStr,c_javaName);//释放局部引用
  14. //构造一个jString对象
  15. char*c_ptr_name="IcomefromNative";
  16. jstringcName=env->NewStringUTF(c_ptr_name);//构造一个jstring对象
  17. env->SetObjectField(obj,nameFieldId,cName);//设置该字段的值
4、回调Java层方法

  1. jstringstr=NULL;
  2. jclassclz=env->FindClass("cc/androidos/jni/JniTest");
  3. //获取clz的构造函数并生成一个对象
  4. jmethodIDctor=env->GetMethodID(clz,"<init>","()V");
  5. jobjectobj=env->NewObject(clz,ctor);
  6. //如果是数组类型,则在类型前加[,如整形数组int[]intArray,则对应类型为[I,整形数组String[]strArray对应为[Ljava/lang/String;
  7. jmethodIDmid=env->GetMethodID(clz,"sayHelloFromJava","(Ljava/lang/String;II[I)I");
  8. if(mid)
  9. {
  10. LOGI("midisget");
  11. jstringstr1=env->NewStringUTF("IamNative");
  12. jintindex1=10;
  13. jintindex2=12;
  14. //env->CallVoidMethod(obj,mid,str1,index1,index2);
  15. //数组类型转换testIntArray能不能不申请内存空间
  16. jintArraytestIntArray=env->NewIntArray(10);
  17. jint*test=newjint[10];
  18. for(inti=0;i<10;++i)
  19. {
  20. *(test+i)=i+100;
  21. }
  22. env->SetIntArrayRegion(testIntArray,0,10,test);
  23. jintjavaIndex=env->CallIntMethod(obj,mid,str1,index1,index2,testIntArray);
  24. LOGI("javaIndex=%d",javaIndex);
  25. delete[]test;
  26. test=NULL;
  27. }

  1. staticvoidevent_callback(inteventId,constchar*description){//主进程回调可以,线程中回调失败。
  2. if(gEventHandle==NULL)
  3. return;
  4. JNIEnv*env;
  5. boolisAttached=false;
  6. if(myVm->GetEnv((void**)&env,JNI_VERSION_1_2)<0){//获取当前的JNIEnv
  7. if(myVm->AttachCurrentThread(&env,NULL)<0)
  8. return;
  9. isAttached=true;
  10. }
  11. jclasscls=env->GetObjectClass(gEventHandle);//获取类对象
  12. if(!cls){
  13. LOGE("EventHandler:failedtogetclassreference");
  14. return;
  15. }
  16. jmethodIDmethodID=env->GetStaticMethodID(cls,"callbackStatic",
  17. "(ILjava/lang/String;)V");//静态方法或成员方法
  18. if(methodID){
  19. jstringcontent=env->NewStringUTF(description);
  20. env->CallVoidMethod(gEventHandle,methodID,eventId,
  21. content);
  22. env->ReleaseStringUTFChars(content,description);
  23. }else{
  24. LOGE("EventHandler:failedtogetthecallbackmethod");
  25. }
  26. if(isAttached)
  27. myVm->DetachCurrentThread();
  28. }

线程中回调
把c/c++中所有线程的创建,由pthread_create函数替换为由Java层的创建线程的函数AndroidRuntime::createJavaThread。
  1. staticpthread_tcreate_thread_callback(constchar*name,void(*start)(void*),void*arg)
  2. {
  3. return(pthread_t)AndroidRuntime::createJavaThread(name,start,arg);
  4. }
  5. staticvoidcheckAndClearExceptionFromCallback(JNIEnv*env,constchar*methodName){//异常检测和排除
  6. if(env->ExceptionCheck()){
  7. LOGE("Anexceptionwasthrownbycallback'%s'.",methodName);
  8. LOGE_EX(env);
  9. env->ExceptionClear();
  10. }
  11. }
  12. staticvoidreceive_callback(unsignedchar*buf,intlen)//回调
  13. {
  14. inti;
  15. JNIEnv*env=AndroidRuntime::getJNIEnv();
  16. jcharArrayarray=env->NewCharArray(len);
  17. jchar*pArray;
  18. if(array==NULL){
  19. LOGE("receive_callback:NewCharArrayerror.");
  20. return;
  21. }
  22. pArray=(jchar*)calloc(len,sizeof(jchar));
  23. if(pArray==NULL){
  24. LOGE("receive_callback:callocerror.");
  25. return;
  26. }
  27. //copybuffertojchararray
  28. for(i=0;i<len;i++)
  29. {
  30. *(pArray+i)=*(buf+i);
  31. }
  32. //copybuffertojcharArray
  33. env->SetCharArrayRegion(array,0,len,pArray);
  34. //invokejavacallbackmethod
  35. env->CallVoidMethod(mCallbacksObj,method_receive,array,len);
  36. //releaseresource
  37. env->DeleteLocalRef(array);
  38. free(pArray);
  39. pArray=NULL;
  40. checkAndClearExceptionFromCallback(env,__FUNCTION__);
  41. }
  42. publicvoidReceive(charbuffer[],intlength){//java层函数
  43. Stringmsg=newString(buffer);
  44. msg="receivedfromjnicallback"+msg;
  45. Log.d("Test",msg);
  46. }

  1. jclasscls=env->GetObjectClass(obj);//获得Java类实例
  2. jmethodIDcallbackID=env->GetMethodID(cls,"callback","(Ljava/lang/String;)V");//或得该回调方法句柄
  3. if(callbackID==NULL)
  4. {
  5. cout<<"getMethodIdisfailed\n"<<endl;
  6. }
  7. jstringnative_desc=env->NewStringUTF("IamNative");
  8. env->CallVoidMethod(obj,callbackID,native_desc);//回调该方法,并且


5、传对象到JNI调用

  1. jclassstu_cls=env->GetObjectClass(obj_stu);//或得Student类引用
  2. if(stu_cls==NULL)
  3. {
  4. cout<<"GetObjectClassfailed\n";
  5. }
  6. //下面这些函数操作,我们都见过的。O(∩_∩)O~
  7. jfieldIDageFieldID=env->GetFieldID(stucls,"age","I");//获得得Student类的属性id
  8. jfieldIDnameFieldID=env->GetFieldID(stucls,"name","Ljava/lang/String;");//获得属性ID
  9. jintage=env->GetIntField(objstu,ageFieldID);//获得属性值
  10. jstringname=(jstring)env->GetObjectField(objstu,nameFieldID);//获得属性值
  11. constchar*c_name=env->GetStringUTFChars(name,NULL);//转换成char*
  12. stringstr_name=c_name;
  13. env->ReleaseStringUTFChars(name,c_name);//释放引用
  14. cout<<"atNativeageis:"<<age<<"#nameis"<<str_name<<endl;
6、与C++互转

jbytearray转c++byte数组

  1. jbyte*arrayBody=env->GetByteArrayElements(data,0);
  2. jsizetheArrayLengthJ=env->GetArrayLength(data);
  3. BYTE*starter=(BYTE*)arrayBody;


jbyteArray 转 c++中的BYTE[]
  1. jbyte*olddata=(jbyte*)env->GetByteArrayElements(strIn,0);
  2. jsizeoldsize=env->GetArrayLength(strIn);
  3. BYTE*bytearr=(BYTE*)olddata;
  4. intlen=(int)oldsize;


C++中的BYTE[]转jbyteArray
  1. jbyte*by=(jbyte*)pData;
  2. jbyteArrayjarray=env->NewByteArray(nOutSize);
  3. env->SetByteArrayRegin(jarray,0,nOutSize,by);


jbyteArray 转 char *
  1. char*data=(char*)env->GetByteArrayElements(strIn,0);


char* 转jstring
  1. jstringWindowsTojstring(JNIEnv*env,char*str_tmp)
  2. {
  3. jstringrtn=0;
  4. intslen=(int)strlen(str_tmp);
  5. unsignedshort*buffer=0;
  6. if(slen==0)
  7. {
  8. rtn=env->NewStringUTF(str_tmp);
  9. }
  10. else
  11. {
  12. intlength=MultiByteToWideChar(CP_ACP,0,(LPCSTR)str_tmp,slen,NULL,0);
  13. buffer=(unsignedshort*)malloc(length*2+1);
  14. if(MultiByteToWideChar(CP_ACP,0,(LPCSTR)str_tmp,slen,(LPWSTR)buffer,length)>0)
  15. {
  16. rtn=env->NewString((jchar*)buffer,length);
  17. }
  18. }
  19. if(buffer)
  20. {
  21. free(buffer);
  22. }
  23. returnrtn;
  24. }


char* jstring互转
  1. JNIEXPORTjstringJNICALLJava_com_explorer_jni_SambaTreeNative_getDetailsBy
  2. (JNIEnv*env,jobjectjobj,jstringpc_server,jstringserver_user,jstringserver_passwd)
  3. {
  4. constchar*pc=env->GetStringUTFChars(pc_server,NULL);
  5. constchar*user=env->GetStringUTFChars(server_user,NULL);
  6. constchar*passwd=env->GetStringUTFChars(server_passwd,NULL);
  7. constchar*details=smbtree::getPara(pc,user,passwd);
  8. jstringjDetails=env->NewStringUTF(details);
  9. returnjDetails;
  10. }
4、Android.mk、Application.mk
1、Android.mk

Android.mk文件是GNU Makefile的一小部分,它用来对Android程序进行编译,Android.mk中的变量都是全局的,解析过程会被定义。

一个Android.mk文件可以编译多个模块,模块包括:APK程序、JAVA库、C\C++应用程序、C\C++静态库、C\C++共享库。

简单实例如下:

  1. LOCAL_PATH:=$(callmy-dir)#表示是当前文件的路径
  2. include$(CLEAR_VARS)#指定让GNUMAKEFILE该脚本为你清除许多LOCAL_XXX变量
  3. LOCAL_MODULE:=helloworld#标识你在Android.mk文件中描述的每个模块
  4. MY_SOURCES:=foo.c#自定义变量
  5. ifneq($(MY_CONFIG_BAR),)
  6. MY_SOURCES+=bar.c
  7. endif
  8. LOCAL_SRC_FILES+=$(MY_SOURCES)#包含将要编译打包进模块中的C或C++源代码文件
  9. include$(BUILD_SHARED_LIBRARY)#根据LOCAL_XXX系列变量中的值,来编译生成共享库(动态链接库)


GNU Make系统变量

变量 描述
CLEAR_VARS 指向一个编译脚本,几乎所有未定义的 LOCAL_XXX 变量都在"Module-description"节中列出。必须在开始一个新模块之前包含这个脚本:include$(CLEAR_VARS),用于重置除LOCAL_PATH变量外的,所有LOCAL_XXX系列变量。
BUILD_SHARED_LIBRARY 指向编译脚本,根据所有的在 LOCAL_XXX 变量把列出的源代码文件编译成一个共享库。
BUILD_STATIC_LIBRARY 一个 BUILD_SHARED_LIBRARY 变量用于编译一个静态库。静态库不会复制到的APK包中,但是能够用于编译共享库。
TARGET_ARCH 目标 CPU平台的名字, 和 android 开放源码中指定的那样。如果是arm,表示要生成 ARM 兼容的指令,与 CPU架构的修订版无关。
TARGET_PLATFORM Android.mk 解析的时候,目标 Android 平台的名字.详情可参考/development/ndk/docs/stable- apis.txt.
TARGET_ARCH_ABI 支持目标平台
TARGET_ABI 目标平台和 ABI 的组合,它事实上被定义成$(TARGET_PLATFORM)-$(TARGET_ARCH_ABI) ,在想要在真实的设备中针对一个特别的目标系统进行测试时,会有用。在默认的情况下,它会是'android-3-arm'。

模块描述变量

变量 描述
LOCAL_PATH 这个变量用于给出当前文件的路径。必须在 Android.mk 的开头定义,可以这样使用:LOCAL_PATH := $(call my-dir) 这个变量不会被$(CLEAR_VARS)清除,因此每
个 Android.mk 只需要定义一次(即使在一个文件中定义了几个模块的情况下)。
LOCAL_MODULE 这是模块的名字,它必须是唯一的,而且不能包含空格。必须在包含任一的$(BUILD_XXXX)脚本之前定义它。模块的名字决定了生成文件的名字。例如,如果一个一个共享库模块的名字是,那么生成文件的名字就是 lib.so。但是,在的 NDK 生成文件中(或者 Android.mk 或者 Application.mk),应该只涉及(引用)有正常名字的其他模块。
LOCAL_SRC_FILES 这是要编译的源代码文件列表。只要列出要传递给编译器的文件,因为编译系统自动计算依赖。注意源代码文件名称都是相对于 LOCAL_PATH的,你可以使用路径部分。
LOCAL_CPP_EXTENSION 这是一个可选变量, 用来指定C++代码文件的扩展名,默认是'.cpp',但是可以改变它。
LOCAL_C_INCLUDES 可选变量,表示头文件的搜索路径。
LOCAL_CFLAGS 可选的编译器选项,在编译 C 代码文件的时候使用。
LOCAL_CXXFLAGS 与 LOCAL_CFLAGS同理,针对 C++源文件。
LOCAL_CPPFLAGS 与 LOCAL_CFLAGS同理,但是对 C 和 C++ source files都适用。
LOCAL_STATIC_LIBRARIES 表示该模块需要使用哪些静态库,以便在编译时进行链接。
LOCAL_SHARED_LIBRARIES 表示模块在运行时要依赖的共享库(动态库),在链接时就需要,以便在生成文件时嵌入其相应的信息。注意:它不会附加列出的模块到编译图,也就是仍然需要在Application.mk 中把它们添加到程序要求的模块中。
LOCAL_LDLIBS 编译模块时要使用的附加的链接器选项。这对于使用‘-l’前缀传递指定库的名字是有用的。
LOCAL_ALLOW_UNDEFINED_SYMBOLS 默认情况下, 在试图编译一个共享库时,任何未定义的引用将导致一个“未定义的符号”错误。
LOCAL_ARM_MODE 默认情况下, arm目标二进制会以 thumb 的形式生成(16 位),你可以通过设置这个变量为 arm如果你希望你的 module 是以 32 位指令的形式。
LOCAL_MODULE_PATH 和 LOCAL_UNSTRIPPED_PATH 在 Android.mk 文件中, 还可以用LOCAL_MODULE_PATH 和LOCAL_UNSTRIPPED_PATH指定最后的目标安装路径.
不同的文件系统路径用以下的宏进行选择:
TARGET_ROOT_OUT:表示根文件系统。
TARGET_OUT:表示 system文件系统。
TARGET_OUT_DATA:表示 data文件系统。
用法如:LOCAL_MODULE_PATH :=$(TARGET_ROOT_OUT)
至于LOCAL_MODULE_PATH 和LOCAL_UNSTRIPPED_PATH的区别,暂时还不清楚。

GNU Make 功能宏

变量 描述
my-dir 返回当前 Android.mk 所在的目录的路径,相对于 NDK 编译系统的顶层。
all-subdir-makefiles 返回一个位于当前'my-dir'路径的子目录中的所有Android.mk的列表。
this-makefile 返回当前Makefile 的路径(即这个函数调用的地方)
parent-makefile 返回调用树中父 Makefile 路径。即包含当前Makefile的Makefile 路径。
grand-parent-makefile 返回调用树中父Makefile的父Makefile的路径

范例:


2、

编译一个简单的APK

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Buildalljavafilesinthejavasubdirectory
  4. LOCAL_SRC_FILES:=$(callall-subdir-java-files)
  5. #NameoftheAPKtobuild
  6. LOCAL_PACKAGE_NAME:=LocalPackage
  7. #TellittobuildanAPK
  8. include$(BUILD_PACKAGE)

编译一个依赖静态.jar文件的APK

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Listofstaticlibrariestoincludeinthepackage
  4. LOCAL_STATIC_JAVA_LIBRARIES:=static-library
  5. #Buildalljavafilesinthejavasubdirectory
  6. LOCAL_SRC_FILES:=$(callall-subdir-java-files)
  7. #NameoftheAPKtobuild
  8. LOCAL_PACKAGE_NAME:=LocalPackage
  9. #TellittobuildanAPK
  10. include$(BUILD_PACKAGE)
  11. 注:LOCAL_STATIC_JAVA_LIBRARIES后面应是你的APK程序所需要的JAVA库的JAR文件名。


编译一个需要platform key签名的APK

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Buildalljavafilesinthejavasubdirectory
  4. LOCAL_SRC_FILES:=$(callall-subdir-java-files)
  5. #NameoftheAPKtobuild
  6. LOCAL_PACKAGE_NAME:=LocalPackage
  7. LOCAL_CERTIFICATE:=platform
  8. #TellittobuildanAPK
  9. include$(BUILD_PACKAGE)
  10. 注:LOCAL_CERTIFICATE后面应该是签名文件的文件名

编译一个需要特殊vendor key签名的APK

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Buildalljavafilesinthejavasubdirectory
  4. LOCAL_SRC_FILES:=$(callall-subdir-java-files)
  5. #NameoftheAPKtobuild
  6. LOCAL_PACKAGE_NAME:=LocalPackage
  7. LOCAL_CERTIFICATE:=vendor/example/certs/app
  8. #TellittobuildanAPK
  9. include$(BUILD_PACKAGE)

装载一个普通的第三方APK

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Modulenameshouldmatchapknametobeinstalled.
  4. LOCAL_MODULE:=LocalModuleName
  5. LOCAL_SRC_FILES:=$(LOCAL_MODULE).apk
  6. LOCAL_MODULE_CLASS:=APPS
  7. LOCAL_MODULE_SUFFIX:=$(COMMON_ANDROID_PACKAGE_SUFFIX)
  8. LOCAL_CERTIFICATE:=platform
  9. include$(BUILD_PREBUILT)

装载需要.so(动态库)的第三方apk

  1. LOCAL_PATH:=$(my-dir)
  2. include$(CLEAR_VARS)
  3. LOCAL_MODULE:=baiduinput_android_v1.1_1000e
  4. LOCAL_SRC_FILES:=$(LOCAL_MODULE).apk
  5. LOCAL_MODULE_CLASS:=APPS
  6. LOCAL_MODULE_SUFFIX:=$(COMMON_ANDROID_PACKAGE_SUFFIX)
  7. LOCAL_CERTIFICATE:=platform
  8. include$(BUILD_PREBUILT)
  9. #################################################################
  10. #######copythelibraryto/system/lib#########################
  11. #################################################################
  12. include$(CLEAR_VARS)
  13. LOCAL_MODULE:=libinputcore.so
  14. LOCAL_MODULE_CLASS:=SHARED_LIBRARIES
  15. LOCAL_MODULE_PATH:=$(TARGET_OUT_SHARED_LIBRARIES)
  16. LOCAL_SRC_FILES:=lib/$(LOCAL_MODULE)
  17. OVERRIDE_BUILD_MODULE_PATH:=$(TARGET_OUT_INTERMEDIATE_LIBRARIES)
  18. include$(BUILD_PREBUILT)

编译一个静态java库

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. #Buildalljavafilesinthejavasubdirectory
  4. LOCAL_SRC_FILES:=$(callall-subdir-java-files)
  5. #Anylibrariesthatthislibrarydependson
  6. LOCAL_JAVA_LIBRARIES:=android.test.runner
  7. #Thenameofthejarfiletocreate
  8. LOCAL_MODULE:=sample
  9. #Buildastaticjarfile.
  10. include$(BUILD_STATIC_JAVA_LIBRARY)
  11. 注:LOCAL_JAVA_LIBRARIES表示生成的java库的jar文件名。

编译C/C++应用程序模板

  1. LOCAL_PATH:=$(callmy-dir)
  2. #include$(CLEAR_VARS)
  3. LOCAL_SRC_FILES:=main.c
  4. LOCAL_MODULE:=test_exe
  5. #LOCAL_C_INCLUDES:=
  6. #LOCAL_STATIC_LIBRARIES:=
  7. #LOCAL_SHARED_LIBRARIES:=
  8. include$(BUILD_EXECUTABLE)
  9. 注:‘:=’是赋值的意思,'+='是追加的意思,‘$’表示引用某变量的值
  10. LOCAL_SRC_FILES中加入源文件路径,LOCAL_C_INCLUDES中加入需要的头文件搜索路径
  11. LOCAL_STATIC_LIBRARIES加入所需要链接的静态库(*.a)的名称,
  12. LOCAL_SHARED_LIBRARIES中加入所需要链接的动态库(*.so)的名称,
  13. LOCAL_MODULE表示模块最终的名称,BUILD_EXECUTABLE表示以一个可执行程序的方式进行编译。
  14. (4)编译C\C++静态库
  15. LOCAL_PATH:=$(callmy-dir)
  16. include$(CLEAR_VARS)
  17. LOCAL_SRC_FILES:=\
  18. helloworld.c
  19. LOCAL_MODULE:=libtest_static
  20. #LOCAL_C_INCLUDES:=
  21. #LOCAL_STATIC_LIBRARIES:=
  22. #LOCAL_SHARED_LIBRARIES:=
  23. include$(BUILD_STATIC_LIBRARY)
  24. 和上面相似,BUILD_STATIC_LIBRARY表示编译一个静态库。

编译C\C++动态库的模板

  1. LOCAL_PATH:=$(callmy-dir)
  2. include$(CLEAR_VARS)
  3. LOCAL_SRC_FILES:=helloworld.c
  4. LOCAL_MODULE:=libtest_shared
  5. TARGET_PRELINK_MODULES:=false
  6. #LOCAL_C_INCLUDES:=
  7. #LOCAL_STATIC_LIBRARIES:=
  8. #LOCAL_SHARED_LIBRARIES:=
  9. include$(BUILD_SHARED_LIBRARY)
  10. 和上面相似,BUILD_SHARED_LIBRARY表示编译一个共享库。
  11. 以上三者的生成结果分别在如下目录中,generic依具体target会变:
  12. out/target/product/generic/obj/APPS
  13. out/target/product/generic/obj/JAVA_LIBRARIES
  14. out/target/product/generic/obj/EXECUTABLE
  15. out/target/product/generic/obj/STATIC_LIBRARY
  16. out/target/product/generic/obj/SHARED_LIBRARY
  17. 每个模块的目标文件夹分别为:
  18. 1)APK程序:XXX_intermediates
  19. 2)JAVA库程序:XXX_intermediates
  20. 这里的XXX
  21. 3)C\C++可执行程序:XXX_intermediates
  22. 4)C\C++静态库:XXX_static_intermediates
  23. 5)C\C++动态库:XXX_shared_intermediates


实例:

  1. LOCAL_PATH:=$(callmy-dir)#项目地址
  2. include$(CLEAR_VARS)#清除变量
  3. LOCAL_MODULE:=libvlcjni#库
  4. #源文件
  5. LOCAL_SRC_FILES:=libvlcjni.clibvlcjni-util.clibvlcjni-track.clibvlcjni-medialist.caout.cvout.clibvlcjni-equalizer.cnative_crash_handler.c
  6. LOCAL_SRC_FILES+=thumbnailer.cpthread-condattr.cpthread-rwlocks.cpthread-once.ceventfd.csem.c
  7. LOCAL_SRC_FILES+=pipe2.c
  8. LOCAL_SRC_FILES+=wchar/wcpcpy.c
  9. LOCAL_SRC_FILES+=wchar/wcpncpy.c
  10. LOCAL_SRC_FILES+=wchar/wcscasecmp.c
  11. LOCAL_SRC_FILES+=wchar/wcscat.c
  12. LOCAL_SRC_FILES+=wchar/wcschr.c
  13. LOCAL_SRC_FILES+=wchar/wcscmp.c
  14. LOCAL_SRC_FILES+=wchar/wcscoll.c
  15. LOCAL_SRC_FILES+=wchar/wcscpy.c
  16. LOCAL_SRC_FILES+=wchar/wcscspn.c
  17. LOCAL_SRC_FILES+=wchar/wcsdup.c
  18. LOCAL_SRC_FILES+=wchar/wcslcat.c
  19. LOCAL_SRC_FILES+=wchar/wcslcpy.c
  20. LOCAL_SRC_FILES+=wchar/wcslen.c
  21. LOCAL_SRC_FILES+=wchar/wcsncasecmp.c
  22. LOCAL_SRC_FILES+=wchar/wcsncat.c
  23. LOCAL_SRC_FILES+=wchar/wcsncmp.c
  24. LOCAL_SRC_FILES+=wchar/wcsncpy.c
  25. LOCAL_SRC_FILES+=wchar/wcsnlen.c
  26. LOCAL_SRC_FILES+=wchar/wcspbrk.c
  27. LOCAL_SRC_FILES+=wchar/wcsrchr.c
  28. LOCAL_SRC_FILES+=wchar/wcsspn.c
  29. LOCAL_SRC_FILES+=wchar/wcsstr.c
  30. LOCAL_SRC_FILES+=wchar/wcstok.c
  31. LOCAL_SRC_FILES+=wchar/wcswidth.c
  32. LOCAL_SRC_FILES+=wchar/wcsxfrm.c
  33. LOCAL_SRC_FILES+=wchar/wmemchr.c
  34. LOCAL_SRC_FILES+=wchar/wmemcmp.c
  35. LOCAL_SRC_FILES+=wchar/wmemcpy.c
  36. LOCAL_SRC_FILES+=wchar/wmemmove.c
  37. LOCAL_SRC_FILES+=wchar/wmemset.c
  38. LOCAL_C_INCLUDES:=$(VLC_SRC_DIR)/include#包含头
  39. ARCH=$(ANDROID_ABI)#变量平台
  40. CPP_STATIC=$(ANDROID_NDK)/sources/cxx-stl/gnu-libstdc++$(CXXSTL)/libs/$(ARCH)/libgnustl_static.a#应用静态库
  41. LOCAL_CFLAGS:=-std=gnu99#编译器标识
  42. ifeq($(ARCH),armeabi)
  43. LOCAL_CFLAGS+=-DHAVE_ARMEABI
  44. #NeededbyARMv6Thumb1(theSystemControlcoprocessor/CP15ismandatoryonARMv6)
  45. #OnnewerARMarchitectureswecanuseThumb2
  46. LOCAL_ARM_MODE:=arm
  47. endif
  48. ifeq($(ARCH),armeabi-v7a)
  49. LOCAL_CFLAGS+=-DHAVE_ARMEABI_V7A
  50. endif
  51. LOCAL_LDLIBS:=-L$(VLC_CONTRIB)/lib\#使用本地库
  52. $(VLC_MODULES)\
  53. $(VLC_BUILD_DIR)/lib/.libs/libvlc.a\
  54. $(VLC_BUILD_DIR)/src/.libs/libvlccore.a\
  55. $(VLC_BUILD_DIR)/compat/.libs/libcompat.a\
  56. -ldl-lz-lm-llog\
  57. -ldvbpsi-lebml-lmatroska-ltag\
  58. -logg-lFLAC-ltheora-lvorbis\
  59. -lmpeg2-la52\
  60. -lavformat-lavcodec-lswscale-lavutil-lpostproc-lgsm-lopenjpeg\
  61. -lliveMedia-lUsageEnvironment-lBasicUsageEnvironment-lgroupsock\
  62. -lspeex-lspeexdsp\
  63. -lxml2-lpng-lgnutls-lgcrypt-lgpg-error\
  64. -lnettle-lhogweed-lgmp\
  65. -lfreetype-liconv-lass-lfribidi-lopus\
  66. -lEGL-lGLESv2-ljpeg\
  67. -ldvdnav-ldvdread-ldvdcss\
  68. $(CPP_STATIC)
  69. include$(BUILD_SHARED_LIBRARY)#编译成动态库
  70. include$(CLEAR_VARS)#清除变量
  71. LOCAL_MODULE:=libiomx-gingerbread
  72. LOCAL_SRC_FILES:=../$(VLC_SRC_DIR)/modules/codec/omxil/iomx.cpp
  73. LOCAL_C_INCLUDES:=$(VLC_SRC_DIR)/modules/codec/omxil$(ANDROID_SYS_HEADERS_GINGERBREAD)/frameworks/base/include$(ANDROID_SYS_HEADERS_GINGERBREAD)/system/core/include
  74. LOCAL_CFLAGS:=-Wno-psabi
  75. LOCAL_LDLIBS:=-L$(ANDROID_LIBS)-lgcc-lstagefright-lmedia-lutils-lbinder
  76. include$(BUILD_SHARED_LIBRARY)
  77. include$(CLEAR_VARS)
  78. LOCAL_MODULE:=libiomx-hc
  79. LOCAL_SRC_FILES:=../$(VLC_SRC_DIR)/modules/codec/omxil/iomx.cpp
  80. LOCAL_C_INCLUDES:=$(VLC_SRC_DIR)/modules/codec/omxil$(ANDROID_SYS_HEADERS_HC)/frameworks/base/include$(ANDROID_SYS_HEADERS_HC)/frameworks/base/native/include$(ANDROID_SYS_HEADERS_HC)/system/core/include$(ANDROID_SYS_HEADERS_HC)/hardware/libhardware/include
  81. LOCAL_CFLAGS:=-Wno-psabi
  82. LOCAL_LDLIBS:=-L$(ANDROID_LIBS)-lgcc-lstagefright-lmedia-lutils-lbinder
  83. include$(BUILD_SHARED_LIBRARY)
  84. include$(CLEAR_VARS)
  85. LOCAL_MODULE:=libiomx-ics
  86. LOCAL_SRC_FILES:=../$(VLC_SRC_DIR)/modules/codec/omxil/iomx.cpp
  87. LOCAL_C_INCLUDES:=$(VLC_SRC_DIR)/modules/codec/omxil$(ANDROID_SYS_HEADERS_ICS)/frameworks/base/include$(ANDROID_SYS_HEADERS_ICS)/frameworks/base/native/include$(ANDROID_SYS_HEADERS_ICS)/system/core/include$(ANDROID_SYS_HEADERS_ICS)/hardware/libhardware/include
  88. LOCAL_CFLAGS:=-Wno-psabi
  89. LOCAL_LDLIBS:=-L$(ANDROID_LIBS)-lgcc-lstagefright-lmedia-lutils-lbinder
  90. include$(BUILD_SHARED_LIBRARY)

2、Application.mk

Application.mk目的是描述在你的应用程序中所需要的模块(即静态库或动态库)。

变量 描述
APP_PROJECT_PATH 这个变量是强制性的,并且会给出应用程序工程的根目录的一个绝对路径。
APP_MODULES 这个变量是可选的,如果没有定义,NDK将由在Android.mk中声明的默认的模块编译,并且包含所有的子文件(makefile文件)如果APP_MODULES定义了,它不许是一个空格分隔的模块列表,这个模块名字被定义在Android.mk文件中的LOCAL_MODULE中。
APP_OPTIM 这个变量是可选的,用来义“release”或"debug"。在编译您的应用程序模块的时候,可以用来改变优先级。
APP_CFLAGS 当编译模块中有任何C文件或者C++文件的时候,C编译器的信号就会被发出。
APP_CXXFLAGS APP_CPPFLAGS的别名,已经考虑在将在未来的版本中废除了
APP_CPPFLAGS 当编译的只有C++源文件的时候,可以通过这个C++编译器来设置
APP_BUILD_SCRIPT 默认情况下,NDK编译系统会在$(APP_PROJECT_PATH)/jni目录下寻找名为Android.mk文件:
$(APP_PROJECT_PATH)/jni/Android.mk
APP_ABI 默认情况下,NDK的编译系统回味"armeabi"ABI生成机器代码。
APP_STL 默认情况下,NDK的编译系统为最小的C++运行时库(/system/lib/libstdc++.so)提供C++头文件。然而,NDK的C++的实现,可以让你使用或着链接在自己的应用程序中。
例如:
APP_STL := stlport_static --> static STLport library
APP_STL := stlport_shared --> shared STLport library
APP_STL := system --> default C++ runtime library

实例:

  1. APP_OPTIM:=release//调试版还是发行版
  2. APP_PLATFORM:=android-8//平台
  3. APP_STL:=gnustl_static//C++运行时库
  4. APP_CPPFLAGS+=-frtti//编译标识
  5. APP_CPPFLAGS+=-fexceptions//编译标识异常
  6. APP_CPPFLAGS+=-DANDROID//编译标识
  7. APP_MODULES:=test//静态模块

JNI内存泄漏

JAVA 编程中的内存泄漏,从泄漏的内存位置角度可以分为两种:JVM 中 Java Heap 的内存泄漏;JVM 内存中 native memory 的内存泄漏。

Java Heap 的内存泄漏:
Java 对象存储在 JVM 进程空间中的 Java Heap 中,Java Heap 可以在 JVM 运行过程中动态变化。如果 Java 对象越来越多,占据 Java Heap 的空间也越来越大,JVM 会在运行时扩充Java Heap 的容量。如果 Java Heap 容量扩充到上限,并且在 GC 后仍然没有足够空间分配新的 Java 对象,便会抛出 out of memory 异常,导致 JVM 进程崩溃。
Java Heap 中 out of memory 异常的出现有两种原因①程序过于庞大,致使过多 Java 对象的同时存在;②程序编写的错误导致 Java Heap 内存泄漏。

JVM 中 native memory 的内存泄漏
从操作系统角度看,JVM 在运行时和其它进程没有本质区别。在系统级别上,它们具有同样的调度机制,同样的内存分配方式,同样的内存格局。
JVM 进程空间中,Java Heap 以外的内存空间称为 JVM 的 native memory。进程的很多资源都是存储在 JVM 的 native memory 中,例如载入的代码映像,线程的堆栈,线程的管理控制块,JVM 的静态数据、全局数据等等。也包括 JNI 程序中 native code 分配到的资源。
在 JVM 运行中,多数进程资源从 native memory 中动态分配。当越来越多的资源在 native memory 中分配,占据越来越多 native memory 空间并且达到 native memory 上限时,JVM 会抛出异常,使 JVM 进程异常退出。而此时 Java Heap 往往还没有达到上限。
多种原因可能导致 JVM 的 native memory 内存泄漏。
例如:
JVM 在运行中过多的线程被创建,并且在同时运行。
JVM 为线程分配的资源就可能耗尽 native memory 的容量。
JNI 编程错误也可能导致 native memory 的内存泄漏。
Native Code 本身的内存泄漏
JNI 编程首先是一门具体的编程语言,或者 C 语言,或者 C++,或者汇编,或者其它 native 的编程语言。每门编程语言环境都实现了自身的内存管理机制。因此,JNI 程序开发者要遵循native 语言本身的内存管理机制,避免造成内存泄漏。以 C 语言为例,当用 malloc() 在进程堆中动态分配内存时,JNI 程序在使用完后,应当调用 free() 将内存释放。总之,所有在native 语言编程中应当注意的内存泄漏规则,在 JNI 编程中依然适应。
Native 语言本身引入的内存泄漏会造成 native memory 的内存,严重情况下会造成 native memory 的 out of memory。
Global Reference 引入的内存泄漏
JNI 编程还要同时遵循 JNI 的规范标准,JVM 附加了 JNI 编程特有的内存管理机制。
JNI 中的 Local Reference 只在 native method 执行时存在,当 native method 执行完后自动失效。这种自动失效,使得对 Local Reference 的使用相对简单,native method 执行完后,它们所引用的 Java 对象的 reference count 会相应减 1。不会造成 Java Heap 中 Java 对象的内存泄漏。
而 Global Reference 对 Java 对象的引用一直有效,因此它们引用的 Java 对象会一直存在 Java Heap 中。程序员在使用 Global Reference 时,需要仔细维护对 Global Reference 的使用。如果一定要使用 Global Reference,务必确保在不用的时候删除。就像在 C 语言中,调用 malloc() 动态分配一块内存之后,调用 free() 释放一样。否则,Global Reference 引用的Java 对象将永远停留在 Java Heap 中,造成 Java Heap 的内存泄漏。
LocalReference 的深入理解
Local Reference 在 native method 执行完成后,会自动被释放,似乎不会造成任何的内存泄漏。但这是错误的。


泄漏实例1:创建大量的 JNI Local Reference
  1. Java代码部分
  2. classTestLocalReference{
  3. privatenativevoidnativeMethod(inti);
  4. publicstaticvoidmain(Stringargs[]){
  5. TestLocalReferencec=newTestLocalReference();
  6. //callthejninativemethod
  7. c.nativeMethod(1000000);
  8. }
  9. static{
  10. //loadthejnilibrary
  11. System.loadLibrary("StaticMethodCall");
  12. }
  13. }
  14. JNI代码,nativeMethod(inti)的C语言实现
  15. #include<stdio.h>
  16. #include<jni.h>
  17. #include"TestLocalReference.h"
  18. JNIEXPORTvoidJNICALLJava_TestLocalReference_nativeMethod
  19. (JNIEnv*env,jobjectobj,jintcount)
  20. {
  21. jinti=0;
  22. jstringstr;
  23. for(;i<count;i++)
  24. str=(*env)->NewStringUTF(env,"0");
  25. }
  26. 运行结果
  27. JVMCI161:FATALERRORinnativemethod:Outofmemorywhenexpanding
  28. localreftablebeyondcapacity
  29. atTestLocalReference.nativeMethod(NativeMethod)
  30. atTestLocalReference.main(TestLocalReference.java:9)

泄漏实例2:建立一个 String 对象,返回给调用函数。
  1. JNI代码,nativeMethod(inti)的C语言实现
  2. #include<stdio.h>
  3. #include<jni.h>
  4. #include"TestLocalReference.h"
  5. jstringCreateStringUTF(JNIEnv*env)
  6. {
  7. return(*env)->NewStringUTF(env,"0");
  8. }
  9. JNIEXPORTvoidJNICALLJava_TestLocalReference_nativeMethod
  10. (JNIEnv*env,jobjectobj,jintcount)
  11. {
  12. jinti=0;
  13. for(;i<count;i++)
  14. {
  15. str=CreateStringUTF(env);
  16. }
  17. }
  18. 运行结果
  19. JVMCI161:FATALERRORinnativemethod:Outofmemorywhenexpandinglocalref
  20. tablebeyondcapacity
  21. atTestLocalReference.nativeMethod(NativeMethod)
  22. atTestLocalReference.main(TestLocalReference.java:9)


编译问题:SLES/OpenSLES.h: No such file or directory
解决方法:ndk-build TARGET_PLATFORM=android-9


编译断点问题:有没有好用的断点工具

解决方法:visualGDB 神器


JNI:http://blog.csdn.net/u013749274/article/category/6258103

分享到:
评论

相关推荐

    Android底层开发实战_Android底层开发实战_android_android开发实战_

    《Android底层开发实战》这...通过阅读《Android底层开发实战》,开发者不仅可以掌握Android底层开发的基础知识,还能学习到如何解决实际开发中遇到的问题,提升系统级优化能力,从而成为一名更全面的Android开发专家。

    从 0 开始带你成为JVM实战高手【完整版】

    通过本教程的学习,您将能够全面掌握JVM的基础知识、实战技巧以及高级进阶内容。无论是初学者还是有一定经验的开发者,都能够从中受益匪浅。希望每位读者都能成为一名真正的JVM实战高手! --- 以上内容基于标题、...

    汪文君高并发编程实战视频资源全集

    │ 高并发编程第二阶段46讲、ClassLoader链接阶段(验证,准备,解析)过程详细介绍.mp4 │ 高并发编程第二阶段47讲、ClassLoader初始化阶段详细介绍clinit.mp4 │ 高并发编程第二阶段48讲、JVM内置三大类加载器...

    实战Java虚拟机

    总之,《实战Java虚拟机》这本书全面涵盖了JVM的各个方面,无论是初学者还是经验丰富的开发者,都能从中受益匪浅,提升自己的Java技术水平。通过深入学习JVM,我们可以优化程序性能,提高代码质量,减少运行时问题,...

    Android JNI加密项目,目前集合DES,AES两种加密算法.zip

    这份资源包内容丰富,涵盖了从基础知识到实战应用的全方位内容,旨在为开发者们提供一个便捷、高效的学习平台。 一、文件手册 资源包中的文件手册部分,详细记录了Android开发的核心知识点和常用技术。无论是初学...

    基于Android的史上最强NDK入门项目实战.zip

    总之,《基于Android的史上最强NDK入门项目实战》旨在帮助你全面掌握NDK开发,让你在Android世界中游刃有余,无论是提升应用性能,还是解决特定问题,都能得心应手。无论你是Android新手还是有一定经验的开发者,这...

    Android高级开发实战 UI、NDK与安全

    5. **性能优化**:学习如何减少布局深度、避免嵌套权重、合理使用视图缓存,以及使用Android Profiler进行UI性能分析。 二、Android NDK开发 1. **C/C++编程**:掌握Android应用中使用C++的基础,如JNI接口、...

    Android底层开发技术实战详解--内核、移植和驱动.(电子工业.王振丽).part3

    书名:《Android底层开发技术实战详解——内核、移植和驱动》(电子工业出版社.王振丽)。本书从底层原理开始讲起,结合真实的案例向读者详细介绍了android内核、移植和驱动开发的整个流程。全书分为19章,依次讲解...

    Android底层开发技术实战详解--内核、移植和驱动.(电子工业.王振丽).part1

    书名:《Android底层开发技术实战详解——内核、移植和驱动》(电子工业出版社.王振丽)。本书从底层原理开始讲起,结合真实的案例向读者详细介绍了android内核、移植和驱动开发的整个流程。全书分为19章,依次讲解...

    android源码开发实战7.03.zip

    总的来说,Android源码开发实战7.03的课程将带领开发者从源码层面全面了解Android系统,提升开发技能,解决实际问题,为创建更高效、更安全的应用提供强大支持。通过深入学习和实践,开发者将能够充分利用Android...

    史上最强NDK入门项目实战.zip

    《史上最强NDK入门项目实战》是一个专门为Android开发者设计...通过这个项目实战,你将能够从理论到实践全面掌握Android NDK开发,为你的移动开发生涯增添更多可能。记得不断实践和探索,NDK的世界充满了无限的可能性。

    android源码开发实战14.13.zip

    在Android源码开发实战中,第14.13章节主要涵盖了Android系统级别的源码分析和应用开发技术。这一部分的学习对于深入理解Android系统的工作原理、优化应用性能以及进行定制化开发至关重要。以下是对这个主题的详细...

    史上最强NDK入门项目实战毕业设计—(包含完整源码可运行).zip

    本项目旨在为初学者提供一个全面的NDK入门实践,通过实际操作帮助理解NDK在Android开发中的应用。 1. **NDK基础知识**: - NDK是什么:NDK是一组工具,让开发者可以在Android应用中嵌入本地代码,提高执行效率和...

    android源码开发实战19.08.zip

    《Android源码开发实战19.08》是针对Android操作系统源码的深入解析与实践指导,涵盖了Android系统从内核到应用层的全方位开发知识。在这个版本中,你将有机会学习到Android系统的最新发展和技术趋势,提升你的...

    android源码开发实战23.01.zip

    综上所述,"android源码开发实战23.01.zip"涵盖了Android开发的多个核心知识点,从系统架构到具体实践,全面地指导开发者掌握Android源码开发的各个环节。通过学习和实践,开发者不仅能提升技能,还能更好地解决实际...

    android源码开发实战4.20.zip

    《Android源码开发实战4.20》是一个深入解析Android操作...以上知识点是《Android源码开发实战4.20》可能涵盖的部分内容,每个点都值得深入研究,通过实践和代码分析,开发者可以全面提升在Android平台上的开发技能。

    android源码开发实战22.11.zip

    9. **JNI与NDK编程**:JNI(Java Native Interface)允许Java代码调用C/C++函数,而NDK则提供了工具集。了解如何使用NDK进行性能敏感的计算或调用硬件特性。 10. **调试与日志系统**:Android的Logcat系统是调试的...

    android源码开发实战20.09.zip

    在这个过程中,开发者将学习到一系列关键知识点,涵盖从系统编译到应用调试的全过程。 1. **Android源码获取与构建**:首先,你需要了解如何从Git仓库获取Android开源项目(AOSP)的源代码。这涉及到设置工作环境,...

Global site tag (gtag.js) - Google Analytics