前面讲到记录到ffmpeg音视频解码的时候,采用的是在主线程中进行操作,这样是不行的,在学习了POSIX多线程操作以后,就可以实现其在子线程中解码了,也可以实现音视频同步了

简单示例

在native实现中,直接调用pthread的多线程方法,这样就可以在JNI层实现多线程,下面是一个简单的实现多线程的例子

#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#include <android/log.h> #include "com_cj5785_posixtest_PosixThread.h" #define LOGI(FORMAT,...) __android_log_print(ANDROID_LOG_INFO, "cj5785",FORMAT,##__VA_ARGS__);
#define LOGE(FORMAT,...) __android_log_print(ANDROID_LOG_ERROR, "cj5785",FORMAT,##__VA_ARGS__); void* thr_func(void *arg)
{
char *buf = (char *)arg;
int i = 0;
for (i = 0; i < 10; ++i) {
LOGI("thread %s --- %d", buf, i);
if(i == 8)
{
pthread_exit((void *)-1);
}
sleep(1);
}
return (void *)0;
} JNIEXPORT void JNICALL Java_com_cj5785_posixtest_PosixThread_pthread
(JNIEnv *env, jobject jobj)
{
//创建多线程
pthread_t tid;
pthread_create(&tid, NULL, thr_func, (void *)"thread-0");
}

运行apk发现,主线程并未被阻塞,多线程开启成功

关于JNIEnv与JavaVM

  • 每个线程都会有自己独立的JNIEnv,因此不建议将主线程的JNIEnv传入到子线程中

    而是应该在子线程中获取到自己的JNIEnv

  • 在Android中,只有一个JavaVM对象

  • 那么就可以通过JavaVM获取到每一个线程关联的JNIEnv

    在JNI中,有两种方式可以得到JavaVM

    • JNI_OnLoad函数中获取,JNI_OnLoad在动态库加载时候就会执行,可以在JNI中实现JNI_OnLoad方法,可以得到验证
    JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved){
    LOGI("%s","JNI_OnLoad run");
    return JNI_VERSION_1_4;
    }
    • (*env)->GetJavaVM(env,&javaVM);,这种方法在创建子线程中调用即可
  • 在得到javaVM以后,就可以将javaVM关联当前线程,就可以获取到当前线程的JNIEnv

(*javaVM)->AttachCurrentThread(javaVM, &env, NULL);
  • 在使用完毕以后必须解除关联
(*javaVM)->DetachCurrentThread(javaVM);

注意:这里有一个大坑,JNIEnv不可作为全局变量保存,而应该使用NewGlobalRef()来新建一个全局引用

Native层本地多线程回调Java函数时env->findClass()失败,这里是由于在子线程中不可以找不到类,这个动作需要在主线程中完成

NDK调试

  • 首先拿到出错地址,在:项目/obj/local/armeadi目录下

    adb logcat | ndk-stack -sym 链接库地址本地位置

    这时候定位到第一个地址,这个地址就是出错地址
  • 使用ndk下的编译链工具查找出错的行数

    adb logcat | arm-linux-androideabi-addr2line -e 链接库地址本地位置+so

完成以上两步就能够定位出错的位置了

通过JNI拿到Java生成的UUID示例

在Java中生成UUID

import java.util.UUID;
public class UUIDUtils {
public static String getUUID() {
return UUID.randomUUID().toString();
}
}

native方法类中的方法

public native void init();
public native void pthread();
public native void destory();

在native实现中得到UUID值

#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#include <android/log.h> #include "com_cj5785_posixtest_PosixThread.h" #define LOGI(FORMAT,...) __android_log_print(4, "cj5785",FORMAT,##__VA_ARGS__);
#define LOGE(FORMAT,...) __android_log_print(6, "cj5785",FORMAT,##__VA_ARGS__); JavaVM *javaVM = NULL;
jclass uuidutils_cls_global = NULL;
jmethodID uuidutils_mid = NULL; //动态库加载时会执行
JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved){
LOGI("%s","JNI_OnLoad");
//javaVM = vm;
return JNI_VERSION_1_4;
} void* thr_func(void *arg)
{
JNIEnv *env = NULL;
//关联
(*javaVM)->AttachCurrentThread(javaVM, &env, NULL);
char *buf = (char *)arg;
int i = 0;
for (i = 0; i < 10; ++i) {
LOGI("%s --- %d", buf, i);
jobject uuid = (*env)->CallStaticObjectMethod(env, uuidutils_cls_global, uuidutils_mid);
const char *uuid_cstr = (*env)->GetStringUTFChars(env, uuid, NULL);
LOGI("%s", uuid_cstr);
(*env)->ReleaseStringUTFChars(env, uuid, uuid_cstr);
}
//解除关联
(*javaVM)->DetachCurrentThread(javaVM);
pthread_exit((void *)0);
} //初始化
JNIEXPORT void JNICALL Java_com_cj5785_posixtest_PosixThread_init
(JNIEnv *env, jobject jobj)
{
//在主线程中获取class以及methodID
jclass uuidutils_jcls = (*env)->FindClass(env, "com/cj5785/posixtest/UUIDUtils");
uuidutils_cls_global = (*env)->NewGlobalRef(env, uuidutils_jcls);
uuidutils_mid = (*env)->GetStaticMethodID(env, uuidutils_jcls, "getUUID", "()Ljava/lang/String;");
} JNIEXPORT void JNICALL Java_com_cj5785_posixtest_PosixThread_pthread
(JNIEnv *env, jobject jobj)
{
(*env)->GetJavaVM(env, &javaVM);
//创建多线程
pthread_t tid;
pthread_create(&tid, NULL, thr_func, (void *)"thread-0");
} //销毁
JNIEXPORT void JNICALL Java_com_cj5785_posixtest_PosixThread_destory
(JNIEnv *env, jobject jobj)
{
(*env)->DeleteGlobalRef(env, uuidutils_cls_global);
}

NDK学习笔记-JNI多线程的更多相关文章

  1. NDK学习笔记-JNI的引用

    JNI中的引用意在告知虚拟机何时回收一个JNI变量 JNI引用变量分为局部引用和全局引用 局部引用 局部引用,通过DeletLocalRef手动释放对象 原因 访问一个很大的Java对象,使用之后还用 ...

  2. NDK学习笔记-JNI的异常处理与缓存策略

    在使用JNI的时候,可能会产生异常,此时就需要对异常进行处理 异常处理 JNI抛出Throwable异常,在Java层可以用Throwable捕捉 而在C只有清空异常这种处理 但如果在JNI中通过Th ...

  3. NDK学习笔记-JNI数据类型和属性方法的访问

    JNI实现了C/C++与Java的相互访问,那么这篇文章就从C/C++访问Java开始说起 native函数说明 每个native函数,都至少有两个参数(JNIEnv *和jclass或jobject ...

  4. NDK学习笔记-JNI开发流程

    JNI(Java Native Interface)Java本地化接口,Java调用C/C++,C/C++调用Java的一套API接口 实现步骤 在Java源文件中编写native方法 public ...

  5. 孙鑫VC学习笔记:多线程编程

    孙鑫VC学习笔记:多线程编程 SkySeraph Dec 11st 2010  HQU Email:zgzhaobo@gmail.com    QQ:452728574 Latest Modified ...

  6. ndk学习20: jni之OnLoad动态注册函数

    一.原理 当在系统中调用System.loadLibrary函数时,该函数会找到对应的动态库, 然后首先试图找到"JNI_OnLoad"函数,如果该函数存在,则调用它 JNI_On ...

  7. android学习笔记----JNI中的c控制java

    面向对象的底层实现 java作为面向对象高级语言,可对现实世界进行建模.和面向过程不同的是面向对象软件的编写不是流程的堆积,而是对业务逻辑的多视角分解和分类.其过程大致为:      1).将知识分解 ...

  8. Android JNI和NDK学习(06)--JNI的数据类型(转)

    本文转自:http://www.cnblogs.com/skywang12345/archive/2013/05/23/3094037.html 本文介绍JNI的数据类型.NDK中关于JNI数据类型的 ...

  9. NDK学习笔记-多线程与生产消费模式

    在做NDK开发的时候,很多情况下都是需要使用多线程的,一方面是提高程序运行效率,另一方面就是防止主线程阻塞 C的多线程 在C语言里,可以通过对于POSIX标准的运用,使得C语言执行多线程 提高程序的执 ...

随机推荐

  1. confluent_kafka消费时内存泄漏

    confluent_kafka测试的内存泄漏的条件 多线程消费 centos6 预测和centos6底层库存在关系. 换用centos7(我是换了7.3)就行了. (起初以为是代码问题,定位问题位置后 ...

  2. P4475 巧克力王国 k-d tree

    思路:\(k-d\ tree\) 提交:2次 错因:\(query\)时有一个\(mx\)误写成\(mn\)窝太菜了. 题解: 先把\(k-d\ tree\)建出来,然后查询时判一下整个矩形是否整体\ ...

  3. [USACO19OPEN]Snakes

    题目链接 题目简介:有n组,每组有若干个蛇的蛇队伍.(也可以理解为n条长度若干的蛇.)我们要用网捕捉,中途可以改变网的大小.目标是浪费空间最小. 解法:首先明确方法:DP.设f[i][t]为捕捉了n条 ...

  4. Excel2013下拉框选择自动填充颜色

    图一写判断公式,图二选择应用范围.

  5. MySQL sql join 算发

    在MySQL中,可以使用批量密钥访问(BKA)连接算法,该算法使用对连接表的索引访问和连接缓冲区. BKA算法支持:内连接,外连接和半连接操作,包括嵌套外连接. BKA的优点:更加高效的表扫描提高了连 ...

  6. (RERERERERERERERERERERE) BZOJ 2746: [HEOI2012]旅行问题

    二次联通门 : BZOJ 2746: [HEOI2012]旅行问题 神TM STL的vector push_back进一个数后取出时就变成了一个很小的负数.. 调不出来了, 不调了 #include ...

  7. 检查errno

    转自 http://blog.csdn.net/todd911/article/details/9132095 很多库函数,特别是那些与操作系统有关的,当执行失败时会通过一个名称为errno的外部变量 ...

  8. mac 安装oh-my-zsh的问题

    安装完,如果想切换回mac原来的bash终端,可以: chsh -s /bin/bash 反之,切换回zsh: chsh -s /bin/zsh

  9. NSString的导出

    字符串的导出,写到某个文件中去 void stringExport(){ NSString *str=@"123456"; //if file not exist will not ...

  10. 理解了这些异常现象才敢说真正懂了TCP协议

    很多人总觉得学习TCP/IP协议没什么用,觉得日常编程开发只需要知道socket接口怎么用就可以了.如果大家定位过线上问题就会知道,实际上并非如此.如果应用在局域网内,且设备一切正常的情况下可能确实如 ...