美文网首页ndk
android使用NDK技术实现7Zip压缩

android使用NDK技术实现7Zip压缩

作者: Coder_Sven | 来源:发表于2020-05-24 00:05 被阅读0次

    下载7zip源代码

    https://sourceforge.net/projects/p7zip/files/

    编译Android可执行文件

    解压源代码,进入 目录/CPP/ANDROID/7zr/jni

    默认编译出armeabi架构,可以根据自己的需要在APPlication.mk中增加/修改,如编译armeabi-v7a和x86:

    APP_ABI:=armeabi 改为 APP_ABI:armeabi-v7a x86

    cmd命令行执行命令ndk-build(笔者用ndk21,ndk16都会报错,最后用ndk12可以正常编译),编译文件输出在CPP/ANDROID/7zr/libs

    默认编译出来的是可执行文件,要在android上使用就需要编译静态库或者动态库

    修改CPP/ANDROID/7zr/jni/Android.mk 最下面的代码

    # Needed since ANDROID 5, these programs run on android-16 (Android 4.1+)
    #PIE是给可执行程序使用的flag(Position-Independent Executable位置无关可执行程序)
    #ndk读取mk文件编译动态库也不需要指定pic
    #LOCAL_CFLAGS += -fPIE
    #LOCAL_LDFLAGS += -fPIE -pie
    
    #include $(BUILD_EXECUTABLE)#可执行文件
    include $(BUILD_SHARED_LIBRARY)#动态库
    #include $(BUILD_STATIC_LIBRARY)#静态库
    

    与编译可执行文件一样操作ndk-build进行编译,可在libs下生成对应的so库。

    1590246026696.png

    在android项目中新建jniLibs文件夹,然后把生成的动态库放进来。

    image-20200523185115600.png

    在app的build.gradle中添加以下配置

    android {
        compileSdkVersion 28
        defaultConfig {
            ......
            externalNativeBuild {
                cmake {
                    cppFlags ""
                    abiFilters 'armeabi-v7a','x86'
                }
            }
        }
       externalNativeBuild {
            cmake {
                path "CMakeLists.txt"
            }
        }
        
    }
    

    编写CMakeLists文件

    image-20200523185959746.png
    
    cmake_minimum_required(VERSION 3.4.1)
    
    
    add_library(
                 native-lib
                 SHARED
                 src/main/cpp/native-lib.cpp )
    
    #设置头文件查找目录
    include_directories(
        src/main/cpp/lib7zr/C
        src/main/cpp/lib7zr/CPP
    )
    
    #设置库查找目录
    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_SOURCE_DIR}/src/main/jniLibs/${CMAKE_ANDROID_ARCH_ABI}")
    
    target_link_libraries(
                           native-lib
                           7zr
                           log )
    

    新建cpp文件夹,将7zip源码拷贝进来

    image-20200523185752091.png

    在app/src/main/cpp下新建native-lib.cpp 文件,编写jni

    [---->native-lib.cpp]

    #include <jni.h>
    #include <string>
    #include <7zTypes.h>
    #include <android/log.h>
    
    #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, "7zr",__VA_ARGS__);
    
    //表示这个函数在别的地方实现
    extern int MY_CDECL main
            (
    #ifndef _WIN32
            int numArgs, char *args[]
    #endif
    );
    
    void strArgs(const char *cmd, int &args, char pString[66][1024]);
    
    extern "C"
    JNIEXPORT jint JNICALL
    Java_com_example_sven_zip_ZipCode_exec(JNIEnv *env, jclass type, jstring cmd_) {
        const char *cmd = env->GetStringUTFChars(cmd_, 0);
        //7zr a /sdcard/7-Zip.7z /sdcard/7-Zip -mx=9
        int numArgs;
        char temp[66][1024] = {0};
        //分割字符串 将值填入变量
        strArgs(cmd, numArgs, temp);
        char *args[] = {0};
        for (int i = 0; i < numArgs; ++i) {
            args[i] = temp[i];
            LOGE("%s", args[i]);
        }
        env->ReleaseStringUTFChars(cmd_, cmd);
        return main(numArgs, args);
    }
    
    void strArgs(const char *cmd, int &numArgs, char argv[66][1024]) {
        //获得字符串长度
        int size = strlen(cmd);
        //argv的两个下标
        int a = 0, b = 0;
        //0 = false
        //记录是否进入空格
        //7zr         a /sdcard/7-Zip.7z /sdcard/7-Zip -mx=9
        // argv[0]="7zr\0"
        //argv[1]="a\0"
        //7zr\0
        int inspace = 0;
        for (int i = 0; i < size; ++i) {
            char c = cmd[i];
            switch (c) {
                case ' ':
                case '\t':
                    if (inspace) {
                        //字符串结束符号
                        argv[a][b++] = '\0';
                        a++;
                        //加入下一个有效字符前 复原
                        b = 0;
                        inspace = 0;
                    }
                    break;
                default:
                    //如果是字符
                    inspace = 1;
                    argv[a][b++] = c;
                    break;
            }
        }
        //7zr a /sdcard/7-Zip.7z /sdcard/7-Zip -mx=9
        //如果最末尾不是空格 就不会进入  case ' ': case '\t': 补上最后一个结束符
    //    if(inspace){}
        if (cmd[size - 1] != ' ' && cmd[size - 1] != '\t') {
            argv[a][b] = '\0';
            a++;
        }
        numArgs = a;
    }
    

    编写一个java类加载native-lib库

    public class ZipCode {
        static {
            System.loadLibrary("native-lib");
        }
    
        public native static int exec(String cmd);
        
    }
    

    在MainActivity测试使用

    public class MainActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
    
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                String[] perms = {Manifest.permission.WRITE_EXTERNAL_STORAGE};
                if (checkSelfPermission(perms[0]) == PackageManager.PERMISSION_DENIED) {
                    requestPermissions(perms, 200);
                }
            }
    
            // 解压   7zr x [压缩文件]  -o[输出目录]
            // 压缩   7zr a [输出文件] [待压缩文件/目录] -mx=9
            File src = new File(Environment.getExternalStorageDirectory(), "7-Zip");
            File out = new File(Environment.getExternalStorageDirectory(), "7-Zip.7z");
            int result = ZipCode.exec("7zr a " + out.getAbsolutePath() + " "
                    + src.getAbsolutePath() + " -mx=9");
            Log.e("Sven", "ZipCode.exec: "+result);
        }
    
    }
    

    注意动态权限。并且保证源文件7-Zip存在,你可以替换成任意一个文件。

    代码地址:

    https://github.com/games2sven/7zip

    相关文章

      网友评论

        本文标题:android使用NDK技术实现7Zip压缩

        本文链接:https://www.haomeiwen.com/subject/xijrihtx.html