ndk10_JNI中链接第三方so库,文件拆分与合并,JNI线程

一、链接第三方so库

  • 1.CMakeList语法链接第三方so库

      cmake_minimum_required(VERSION 3.4.1)
      
      #查找系统提供的库
      find_library( log-lib
                    log )
        
      #每链接一个so库都需要执行设置以下四步
      #第一步:设置so库路径 
      set(my_lib_path ${CMAKE_SOURCE_DIR}/libs)
      #第二步:将第三方库作为动态库引用
      add_library( native-lib
                   SHARED
                   IMPORTED )
                   
      #第三步:指名第三方库的绝对路径
          #my_lib_path 当前CMakeList所在路径
          #ANDROID_ABI 动态链接的指令
          #libnative-lib.so 库名
      set_target_properties( native-lib
                             PROPERTIES IMPORTED_LOCATION
                             ${my_lib_path}/${ANDROID_ABI}/libnative-lib.so )
      
      
      add_library( libTest # 设置库的名字.   
                   SHARED  # 设置为共享库
                   src/main/cpp/native-lib.c  # 源文件路径)
      
      
      #链接需要使用的so库
      target_link_libraries( libTest   #本项目编译的so库  
                             native-lib    # 第四步:链接第三方的so库 
                             ${log-lib}  #链接系统的log库)
    

第三方库的头文件中的方法需要使用extern 修饰

    #ifndef ASLIBTEST_NATIVE_HEAD_H
    #define ASLIBTEST_NATIVE_HEAD_H
    
    //申明外部 函数 外部属性
    extern int addTest(int a, int b);
    
    #endif //ASLIBTEST_NATIVE_HEAD_H
  • 2.android.mk语法链接第三方so库
    LOCAL_TATH :=$(call my-dir)
    
    include $(CLEAR_VARS)   #第一步:引入第三方so库
    LOCAL_MODULE :=fmod     #第二步:第三方库的名称
    LOCAL_SRC_FILES := libfmod.so  #第三步:第三方so库的绝对路径(当前为android.mk所在目录)
    include $(PREBUILT_SHARED_LIBRARY)  #第四步:引入的形式 生成动态库
    
    #引入多个第三方so库
    include $(CLEAR_VARS)
    LOCAL_MODULE :=fmodL
    LOCAL_SRC_FILES := libfmodL.so
    include $(PREBUILT_SHARED_LIBRARY)
    
    
    include $(CLEAR_VARS)
    LOCAL_MODULE := hello         #自己的so库名
    LOCAL_SRC_FILES := hello.c    #要编译的源文件
    LOCAL_SHARED_LIBRARIES : = fmod fmodL   #第五步:链接第三方库
    LOCAL_LDLIBS :=-llog    #引入log库
    include $(PREBUILT_SHARED_LIBRARY)  #指定生成动态库

二、文件拆分与合并

  • 1.java层native方法

      public class FileUtils {
          //文件拆分
          public static native void diff(String path, String pattern_Path, int file_num);
          //文件合并
          public static native void patch(String merger_path, String pattern_Path, int file_num);
      }
    
  • 2.调用native

    /* 文件拆分 (记得添加文件读写权限)
    * path 需要拆分的源文件路径
    * pattern_Path 拆分后的子文件路径和命名
    * %d C语言的int占位符 通过java层传递占位符过去C中就可以用int表示子文件1234
    * 4 : 需要拆分的子文件个数
    * */
    public void diff(View v) {
        String path = SD_CARD_PATH + File.separatorChar + "Vibrato.mp4";  //源文件路径
        String pattern_Path = SD_CARD_PATH + File.separatorChar + "Vibrato_%d.mp4"; //
        FileUtils.diff(path,pattern_Path,4);
    }

    /*
    * 文件合并
    * */
    public void patch(View v) {
        String path = SD_CARD_PATH + File.separatorChar + "Vibrato_merger.mp4";
        String pattern_Path = SD_CARD_PATH + File.separatorChar + "Vibrato_%d.mp4";
        FileUtils.patch(path,pattern_Path,4);
    }
  • 3.文件拆分C实现

      #include <stdio.h>
      #include <malloc.h>
    
      //获取文件大小
      long get_file_size(const char *path) {
          FILE *fp = fopen(path, "rb"); //打开一个文件, 文件必须存在,只运行读
          fseek(fp, 0, SEEK_END);
          long ret = ftell(fp);
          fclose(fp);
          return ret;
      }
    
      JNIEXPORT void JNICALL native_diff
              (JNIEnv *env, jclass clazz, jstring path, jstring pattern_Path, jint file_num) {
          LOGI("JNI native diff begin");
          const char *path_Str = (*env)->GetStringUTFChars(env, path, NULL);
          const char *pattern_Path_str = (*env)->GetStringUTFChars(env, pattern_Path, NULL);
      
          //申请二维字符数据, 存放子文件名
          char **patches = (char **) malloc(sizeof(char *) * file_num);
      
          int i = 0;
          for (; i < file_num; i++) {
              //每个文件名申请地址
              LOGI("char = %d char * = %d", sizeof(char), sizeof(char *));
              patches[i] = (char *) malloc(sizeof(char) * 100);
              // 需要分割的文件 Vibrato.mp4
              // 每个子文件名称 Vibrato_n.mp4
              sprintf(patches[i], pattern_Path_str, i);// 格式化文件名
              LOGI("patch path : %s", patches[i]);
          }
      
          int fileSize = get_file_size(path_Str);//获取文件大小
          FILE *fpr = fopen(path_Str, "rb");
          /*
           * 1.判断文件大小能够被 file_num整除
           * 2.能整除就平分
           * 3.不能整除就先分 file_num -1
           * */
      
          if (fileSize % file_num == 0) { //能够被整除
              int part = fileSize / file_num;
              for (int i = 0; i < file_num; i++) {
                  FILE *fpw = fopen(patches[i], "wb");//文件已经存在 就删除,只运行写
                  for (int j = 0; j < part; j++) {
                      fputc(fgetc(fpr), fpw);
                  }
                  fclose(fpw);
              }
          } else { //不能被整除
              int part = fileSize / (file_num - 1);
              for (int i = 0; i < file_num - 1; i++) {
                  FILE *fpw = fopen(patches[i], "wb");//文件已经存在 就删除,只运行写
                  for (int j = 0; j < part; j++) {
                      fputc(fgetc(fpr), fpw);
                  }
                  fclose(fpw);
              }
      
              FILE *fpw = fopen(patches[file_num - 1], "wb");
              for (int i = 0; i < fileSize % (file_num - 1); i++) {
                  fputc(fgetc(fpr), fpw);
              }
              fclose(fpw);
          }
          fclose(fpr);
          for (int i = 0; i < file_num; i++) {
              free(patches[i]);
          }
          free(patches);
          (*env)->ReleaseStringUTFChars(env, path, path_Str);
          (*env)->ReleaseStringUTFChars(env, pattern_Path, pattern_Path_str);
      }
    
  • 4.文件合并C实现

    JNIEXPORT void JNICALL native_patch
            (JNIEnv *env, jclass clazz, jstring merge_path, jstring pattern_Path, jint file_num) {
        LOGI("JNI native patch begin");
        const char *path_Str = (*env)->GetStringUTFChars(env, merge_path, NULL);
        const char *pattern_Path_str = (*env)->GetStringUTFChars(env, pattern_Path, NULL);
    
        //申请二维字符数据, 存放子文件名
        char **patches = (char **) malloc(sizeof(char *) * file_num);
    
        int i = 0;
        for (; i < file_num; i++) {
            //每个文件名申请地址
    //        LOGI("char = %d char * = %d", sizeof(char), sizeof(char *));
            patches[i] = (char *) malloc(sizeof(char) * 100);
            // 需要分割的文件 Vibrato.mp4
            // 每个子文件名称 Vibrato_n.mp4
            sprintf(patches[i], pattern_Path_str, i);// 格式化文件名
            LOGI("patch path : %s", patches[i]);
        }
    
        FILE *fpw = fopen(path_Str, "wb");
    
        for (int i = 0; i < file_num; i++) {
            int filesize = get_file_size(patches[i]);
            FILE *fpr = fopen(patches[i], "rb");
            for (int j = 0; j < filesize; j++) {
                fputc(fgetc(fpr), fpw);
            }
            fclose(fpr);
        }
        fclose(fpw);
    
        for (int i = 0; i < file_num; i++) {
            free(patches[i]); //每一个malloc 对应一个free
        }
        free(patches);
        (*env)->ReleaseStringUTFChars(env, merge_path, path_Str);
        (*env)->ReleaseStringUTFChars(env, pattern_Path, pattern_Path_str);
    }

三、JNI开设中开设子线程

jni方法和java方法运行在同一个线程,

每个进程有一个jvm* jvmex 每个线程对应一个env

  • 1.在java的MainActivity中举例定义native方法

      @Override
      protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
          setContentView(R.layout.activity_main);
    
          setJniEnv();  //这个方法让jni保持JVM和class对象
          newJniThread();  //创建线程
    
      }
    
      private native void newJniThread(); //JNI中创建线程
    
      private native void setJniEnv();  //通过
    
      //JNI回调java方法1 子线程
      private static void formJni( int i) {
          Log.d("MainActivity","回调 jni : " +i);
      };
      //JNI回调java方法2 子线程
      private void form_JNI_Again(int i) {
          Log.v("MainActivity","回调_JNI_Again : "+i);
      }
    
  • 2.jni中C实现开设新线程

      #include <pthread.h>
      
      JavaVM *g_jvm = NULL;  //将jvm保存为全局变量了
      jobject g_obj = NULL;  //当前的jobject
      
      void *thread_fun(void *arg) {
      
          JNIEnv *env;
          jclass cls;
          jmethodID mid, mid1;
          
          //通过g_jvm 生成了一个新线程的env
          if ((*g_jvm)->AttachCurrentThread(g_jvm, &env, NULL) != JNI_OK) {
              LOGI("%s AttachCurrentThread error failed ", __FUNCTION__);
              return NULL;
          }
      
          //寻找新线程的jclass
          cls = (*env)->GetObjectClass(env, g_obj);
          if (cls == NULL) {
              LOGI("findClass error....");
              goto error;
          }
          //回调java方法
          LOGI("call back begin");
          mid = (*env)->GetStaticMethodID(env, cls, "formJni", "(I)V");
          if (mid == NULL) {
              LOGI("GetStaticMethodID error....");
              goto error;
          }
      
          (*env)->CallStaticVoidMethod(env, cls, mid, (int) arg);
      
          mid1 = (*env)->GetMethodID(env, cls, "form_JNI_Again", "(I)V");
          if (mid1 == NULL) {
              LOGI("GetMethodID error....");
              goto error;
          }
          (*env)->CallVoidMethod(env, g_obj, mid1, (int) arg);
      
          error:
          if ((*g_jvm)->DetachCurrentThread(g_jvm) != JNI_OK) {
              LOGI("%s DetachCurrentThread error failed ", __FUNCTION__);
          }
          pthread_exit(0);
      }
      
      JNIEXPORT void JNICALL native_newThead
              (JNIEnv *env, jclass clazz) {
          LOGI("newThread begin开启新线程");
          int i;
          pthread_t pt[5];
      
          for (i = 0; i < 5; i++) { //开五个线程
              pthread_create(&pt[i], NULL, &thread_fun, (void *) i);
          }
      
      }
      
      JNIEXPORT void JNICALL native_setJniEnv
              (JNIEnv *env, jobject obj) {
          LOGI("native_setJniEnv");
          //保存JVM
          (*env)->GetJavaVM(env, &g_jvm);
          //保持actvity对象(这是一个强引用 用完需要释放)
          g_obj = (*env)->NewGlobalRef(env, obj);
      
      }
    
    
      //释放强引用避免内存泄漏
      //在activity中写一个native方法,在它onDestroy的时候触发
      JNIEXPORT void JNICALL releseClass(JNIEnv *env, jclass clazz){
          (*env)->DeleteGlobalRef(env,g_obj)
      }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 197,737评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,103评论 2 375
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 144,710评论 0 326
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,909评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,794评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,557评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,939评论 3 388
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,572评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,852评论 1 293
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,871评论 2 314
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,692评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,490评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,939评论 3 300
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,114评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,409评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,971评论 2 343
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,176评论 2 339

推荐阅读更多精彩内容