第十二节(文件拆分)

实现流程:利用ndk动态注册方式将sd卡上面指定的视频进行拆分(暂且分四部分),拆分完成之后再进行合并,验证合并完成之后的视频能否正常播放。

先熟悉几个操作模拟器的命令:
adb shell、ls、ls -l、cd
adb shell回车后会出现一个#,这时候输入ls可以看到所有文件夹名,输入ls -l可以看到详细的文件夹,打开文件夹输入 cd 文件夹名,返回上一层输入cd ..

开始写代码

1.准备一个视频文件xiaoxingyun.mkv(大概90M的样子),利用push命令放入sd卡

hhh:Desktop huozhenpeng$ adb push xiaoxingyun.mkv /sdcard/

push 完成:xiaoxingyun.mkv: 1 file pushed. 12.2 MB/s (95102663 bytes in 7.448s)
利用ls -l命令查看下:

image.png

2.先来进行文件的拆分
在MainActivity中定义一个native方法:

public  native  void spliteByJNI(String sourceFilePath,String splitFilePath,int num);

利用javap命令看下该方法的签名,动态注册需要用到

  public native void spliteByJNI(java.lang.String, java.lang.String, int);
    descriptor: (Ljava/lang/String;Ljava/lang/String;I)V

实现代码:
native-lib.cpp

#include <jni.h>
#include <string>
#include<stdlib.h>


#include <android/log.h>
#include <assert.h>

#define TAG "HZP_JNI"
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__)
// 获取数组的大小
#define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0])))


//获取文件大小
long getFileSize(const char *path)
{
    FILE *fp=fopen(path,"rb");
    fseek(fp,0,SEEK_END);
    long size=ftell(fp);
    fclose(fp);
    return  size;
}

JNIEXPORT void JNICALL native_spliteByJNI
        (JNIEnv *env, jclass clazz,jstring sourcepath,jstring split_path,jint num)
{

    LOGI("JNI begin 动态注册的方法 ");
    LOGI("JNI begin 拆分开始 ");
    //首先获取到源文件路径和要拆分成的文件的路径
    const char * source_p=env->GetStringUTFChars(sourcepath,NULL);
    const char * split_p=env->GetStringUTFChars(split_path,NULL);
    //申请一个二维的char数组,用于存放拆分成的文件的名字
    char ** patches=(char **)malloc(sizeof(char *)*num);
    //循环为每个文件名申请地址
    for(int i=0;i<num;i++)
    {
        patches[i]=(char *)malloc(sizeof(char)*100);//我们认为每个文件的名字世超不过100个字符的
        //把要拆分成的文件的名字进行值替换(替换其中的%d)xiaoxingyun_%d.mkv
        sprintf(patches[i],split_p,i);
    }
    //进行文件拆分,先获取源文件的大小
    long fileSize=getFileSize(source_p);
    //源文件
    FILE *fp=fopen(source_p,"rb");
    /**
     * 拆分逻辑:如果可以恰好平分,则全部是相等的,如果不可以,前面的(num-1)个是相等的额,剩余部分给第num个
     */
    if(fileSize%num==0)
    {
        int part=fileSize/num;
        for(int i=0;i<num;i++)
        {
            FILE *fps=fopen(patches[i],"wb");
            //开始写入
            for(int j=0;j<part;j++)
            {
                fputc(fgetc(fp),fps);
            }
            fclose(fps);
        }


    } else{
        int part=fileSize/(num-1);
        for(int i=0;i<num-1;i++)
        {
            LOGI("JNI  ",env->NewStringUTF(patches[i]));
            FILE *fps=fopen(patches[i],"wb");
            //开始写入
            for(int j=0;j<part;j++)
            {
                fputc(fgetc(fp),fps);
            }
            fclose(fps);
        }
        //写入最后一个
        FILE*fpl=fopen(patches[num-1],"wb");
        for(int j=0;j<fileSize%(num-1);j++)
        {
            fputc(fgetc(fp),fpl);
        }
        fclose(fpl);
    }
    fclose(fp);
    for(int i=0;i<num;i++)
    {
        free(patches[i]);
    }
    free(patches);
    env->ReleaseStringUTFChars(sourcepath,source_p);
    env->ReleaseStringUTFChars(split_path,split_p);
    LOGI("JNI begin 拆分结束 ");
}


const JNINativeMethod gMethods[] = {
        {
                "spliteByJNI","(Ljava/lang/String;Ljava/lang/String;I)V",(void*)native_spliteByJNI
        }
};


int registerNatives(JNIEnv* engv)
{
    LOGI("registerNatives begin");
    jclass  clazz;
    clazz = engv -> FindClass("com/example/huozhenpeng/myapplication/MainActivity");

    if (clazz == NULL) {
        LOGI("clazz is null");
        return JNI_FALSE;
    }

    if (engv ->RegisterNatives( clazz, gMethods, NELEM(gMethods)) < 0) {
        LOGI("RegisterNatives error");
        return JNI_FALSE;
    }

    return JNI_TRUE;
}


JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved)
{

    LOGI("jni_OnLoad begin");

    JNIEnv* env = NULL;
    jint result = -1;

    if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {
        LOGI("ERROR: GetEnv failed\n");
        return -1;
    }
    assert(env != NULL);

    registerNatives(env);

    return JNI_VERSION_1_4;
}

写代码过程中可能会遇到bug,ndk的调试可以用lldb命令:用过xcode的应该比较熟悉

image.png

MainActivity.java

package com.example.huozhenpeng.myapplication;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.os.Environment;
import android.support.v4.app.ActivityCompat;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        requestCameraAccess();
        tv_split= (TextView) findViewById(R.id.tv_split);
        tv_merge= (TextView) findViewById(R.id.tv_merge);
        tv_split.setOnClickListener(this);
        tv_merge.setOnClickListener(this);



        sourceFilePath= Environment.getExternalStorageDirectory().getAbsolutePath().toString()+"/xiaoxingyun.mkv";
        splitFilePath=Environment.getExternalStorageDirectory().getAbsolutePath().toString()+"/xiaoxingyun_%d.mkv";

    }

    private int REQUEST_CAMERA_CODE=1;
    private void requestCameraAccess() {

        ActivityCompat.requestPermissions(this, new String[]{ Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CAMERA_CODE);

    }


    private TextView tv_split;
    private TextView tv_merge;

    private String sourceFilePath;
    private String splitFilePath;

    @Override
    public void onClick(View v) {
        switch (v.getId())
        {
            case R.id.tv_merge:
                break;
            case R.id.tv_split:
                spliteByJNI(sourceFilePath,splitFilePath,4);
                break;
        }
    }

    public  native  void spliteByJNI(String sourceFilePath,String splitFilePath,int num);




}

布局文件就不写了
看下结果:

image.png
image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 156,907评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,546评论 1 289
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,705评论 0 238
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,624评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,940评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,371评论 1 210
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,672评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,396评论 0 195
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,069评论 1 238
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,350评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,876评论 1 256
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,243评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,847评论 3 231
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,004评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,755评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,378评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,266评论 2 259

推荐阅读更多精彩内容