700字范文,内容丰富有趣,生活中的好帮手!
700字范文 > 【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( jni 中 main 函

【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( jni 中 main 函

时间:2019-03-30 20:29:20

相关推荐

【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( jni 中 main 函

文章目录

一、JNI 中 main 函数声明二、命令字符串切割并传入 main 函数三、完整代码示例1、完整 jni 代码2、完整 java 代码3、执行结果四、参考资料

前置博客 :

【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( 修改 7zr 交叉编译脚本 Android.mk | 交叉编译 lib7zr.so 动态库 )【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( 拷贝 lib7zr.so 动态库到 Android Studio 工程 | 配置 build.gradle 构建脚本 )【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( 拷贝 lib7zr.so 动态库头文件到 Android 工程中 | 配置 CMakeLists.txt 构建脚本 )【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( 测试 lib7zr.so 动态库调用 )

一、JNI 中 main 函数声明

使用 7zr 可执行程序处理压缩文件时时 , 调用的是其主函数 , CPP\7zip\UI\Console\MainAr.cpp 中的 main 函数 , 传入7z a outputFile inputFile -mx=compressDegree -tcompressType压缩命令 , 或7z x [输入文件] -o[输出目录]解压命令 , 都是使用该主函数接收相关参数 ;

int MY_CDECL main(#ifndef _WIN32int numArgs, char *args[]#endif)

int numArgs 参数表示字符串个数 ;

7zr a files.7z files -mx=9 -t7z命令中 , 有 666 个字符串 , 由 555 个空格隔开 ;

char *args[] 是 指针数组 , 数组中的元素是 char * 类型的指针 , 就是字符串 , 这是个字符串数组 ;

7zr 程序中的主要的头文件是 7zTypes.h , 该头文件中 声明了主要的 类型 和 函数 ; 引入该头文件 ;

#include <7zTypes.h>

声明外部函数 :

// 表示该函数在其它代码中实现// 这是 CPP\7zip\UI\Console\MainAr.cpp 中的 main 函数extern int MY_CDECL main(#ifndef _WIN32int numArgs, char *args[]#endif);

点击声明左侧的双向箭头按钮 , 可以跳转到 MainAr.cpp 的 main 函数位置 ;

跳转位置 :

二、命令字符串切割并传入 main 函数

调用 main 函数 , 需要传入对应的参数 , 分别是

int numArgs 字符串个数 ;

char *args[] 字符串数组 ; 指针数组 , 每个数组元素中都有一个 char * 指针元素 , 指向字符串 ;

int MY_CDECL main(#ifndef _WIN32int numArgs, char *args[]#endif)

从 Java 传入 C 的指令如下 :

7zr a files.7z files -mx=9 -t7z

使用空格将上述指令切割成 666 个字符串 , 然后传入 main 函数 ;

字符串切割过程 :

// 命令示例 : 7zr a files.7z files -mx=9 -t7z// 参数个数int argCount = 0;// 存放多个字符串, 最多 20 个字符串 , 每个最多 1024 个字符char argArray[20][1024] = {0};//分割字符串 将值填入变量// 获取 cmd_java 字符串长度int cmd_size = strlen(cmd_java);// 二维数组 循环控制变量, 第一个是字符串数组 , 第二个是字符串中的字符数组int str_index = 0, char_index = 0;// 标记字符是否是非空字符, tab, 如果是则设置 1 , 如果不是设置 0// 开始时默认 0, 不是空格int isChar = 0;// 逐个字节遍历 字符for(int i = 0; i < cmd_size; i ++){// 获取一个字符char c = cmd_java[i];//LOGI("遍历 %d . %c , cmd_size = %d", i, c, cmd_size);switch (c) {case ' ': // 判断是否是空格case '\t': // 判断是否是 TAB 空格if(isChar){// 如果上一个字符不是空格 , 则需要结束当前的字符串argArray[str_index][char_index++] = '\0';// 字符串索引自增 1str_index ++;// 字符串内字符索引归零char_index = 0;// 设置当前的字符为空格标志位 1isChar = 0;}else{// 如果之前是空格, 那么现在也是空格 ,// 说明命令中有多个空格 , 此处不做任何处理}break;default:isChar = 1;// 将当前字符放入数组中argArray[str_index][char_index++] = c;break;}}// 如果最后一位不是空格 , 则需要手动将最后一个字符串写入到数组中if (cmd_java[cmd_size - 1] != ' ' && cmd_java[cmd_size - 1] != '\t') {// 如果上一个字符不是空格 , 则需要结束当前的字符串argArray[str_index][char_index++] = '\0';// 字符串索引自增 1str_index ++;}// 统计字符串个数argCount = str_index;// 拼装字符串数组char *args[] = {0};for (int i = 0; i < argCount; ++i) {args[i] = argArray[i];// 打印字符串数组LOGI("%d . %s", i, args[i]);}

最终的字符串个数是 argCount , 字符串数组 args ;

将这两个参数传入 main 函数即可 ;

三、完整代码示例

1、完整 jni 代码

完整 jni 层 C++ 代码如下 :

#include <jni.h>#include <string>#include <7zTypes.h>#include "logging_macros.h"// 表示该函数在其它代码中实现// 这是 CPP\7zip\UI\Console\MainAr.cpp 中的 main 函数extern int MY_CDECL main(#ifndef _WIN32int numArgs, char *args[]#endif);extern "C"JNIEXPORT void JNICALLJava_kim_hsl_a7_1zip_MainActivity_executeCmd(JNIEnv* env, jobject thiz, jstring cmd) {// 将 Java 字符串转为 C 字符串const char *cmd_java = env->GetStringUTFChars(cmd, 0);LOGI("jni 中处理压缩文件命令 : %s", cmd_java);// 命令示例 : 7zr a files.7z files -mx=9 -t7z// 参数个数int argCount = 0;// 存放多个字符串, 最多 20 个字符串 , 每个最多 1024 个字符char argArray[20][1024] = {0};//分割字符串 将值填入变量// 获取 cmd_java 字符串长度int cmd_size = strlen(cmd_java);// 二维数组 循环控制变量, 第一个是字符串数组 , 第二个是字符串中的字符数组int str_index = 0, char_index = 0;// 标记字符是否是非空字符, tab, 如果是则设置 1 , 如果不是设置 0// 开始时默认 0, 不是空格int isChar = 0;// 逐个字节遍历 字符for(int i = 0; i < cmd_size; i ++){// 获取一个字符char c = cmd_java[i];//LOGI("遍历 %d . %c , cmd_size = %d", i, c, cmd_size);switch (c) {case ' ': // 判断是否是空格case '\t': // 判断是否是 TAB 空格if(isChar){// 如果上一个字符不是空格 , 则需要结束当前的字符串argArray[str_index][char_index++] = '\0';// 字符串索引自增 1str_index ++;// 字符串内字符索引归零char_index = 0;// 设置当前的字符为空格标志位 1isChar = 0;}else{// 如果之前是空格, 那么现在也是空格 ,// 说明命令中有多个空格 , 此处不做任何处理}break;default:isChar = 1;// 将当前字符放入数组中argArray[str_index][char_index++] = c;break;}}// 如果最后一位不是空格 , 则需要手动将最后一个字符串写入到数组中if (cmd_java[cmd_size - 1] != ' ' && cmd_java[cmd_size - 1] != '\t') {// 如果上一个字符不是空格 , 则需要结束当前的字符串argArray[str_index][char_index++] = '\0';// 字符串索引自增 1str_index ++;}// 统计字符串个数argCount = str_index;// 拼装字符串数组char *args[] = {0};for (int i = 0; i < argCount; ++i) {args[i] = argArray[i];// 打印字符串数组LOGI("%d . %s", i, args[i]);}// 量参数传入 main 函数main(argCount, args);// 释放 Java 字符串以及 C 字符串env->ReleaseStringUTFChars(cmd, cmd_java);LOGI("7zr 命令执行完毕 !");}

2、完整 java 代码

package kim.hsl.a7_zipimport android.os.Buildimport android.os.Bundleimport android.util.Logimport androidx.appcompat.app.AppCompatActivityimport java.io.*class MainActivity : AppCompatActivity() {companion object {val TAG = "MainActivity"init {System.loadLibrary("native-lib")}}override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)copy7zr()compress7z()uncompress7z()compress7zJni()}/*** 将 7zr 文件拷贝到应用私有目录*/fun copy7zr() {Log.i(TAG, "开始拷贝 7zr 文件")// /data/user/0/kim.hsl.a7_zip/files/7zrvar exeFile = File(filesDir, "7zr")Log.i(TAG, "filesDir = ${filesDir.absolutePath} , exeFile = ${exeFile.absolutePath}")// 查看该文件是否存在, 如果存在设置该文件可执行// 如果不存在 , 拷贝文件if (exeFile.exists()) {exeFile.setExecutable(true)Log.i(TAG, "内置存储空间存在该 /data/user/0/kim.hsl.a7_zip/files/7zr 文件")return} else {Log.i(TAG, "内置存储空间不存在 7zr 可执行文件 , 开始拷贝文件")}// 如果不存在 , 拷贝文件var inputStream: InputStream = assets.open("libs/arm64-v8a/7zr")// /data/user/0/kim.hsl.a7_zip/files/7zrvar fileOutputStream: FileOutputStream = FileOutputStream(exeFile)Log.i(TAG, "Build.CPU_ABI = ${Build.CPU_ABI}")// 不同 CPU 架构拷贝不同的可执行程序if (Build.CPU_ABI.startsWith("armeabi-v7a")) {inputStream = assets.open("libs/armeabi-v7a/7zr")} else if (Build.CPU_ABI.startsWith("arm64-v8a")) {inputStream = assets.open("libs/arm64-v8a/7zr")} else if (Build.CPU_ABI.startsWith("x86")) {inputStream = assets.open("libs/x86/7zr")} else if (Build.CPU_ABI.startsWith("x86_64")) {inputStream = assets.open("libs/x86_64/7zr")}// 拷贝文件var buffer: ByteArray = ByteArray(1024)var readCount = inputStream.read(buffer);while (readCount != -1) {fileOutputStream.write(buffer)readCount = inputStream.read(buffer);}fileOutputStream.flush()fileOutputStream.close()Log.i(TAG, "拷贝 7zr 文件结束")}/*** 使用 7zr 进行压缩*/fun compress7z() {// /data/user/0/kim.hsl.a7_zip/files/7zrvar exeFile = File(filesDir, "7zr")// 执行前赋予可执行权限exeFile.setExecutable(true)var file_7z = File("${filesDir.absolutePath}/files.7z")if(file_7z.exists()){file_7z.delete()}var cmd = "${exeFile.absolutePath} a ${filesDir.absolutePath}/files.7z ${filesDir.absolutePath} -mx=9 -t7z"Log.i(TAG, "压缩命令 : $cmd")var process: Process = Runtime.getRuntime().exec(cmd)// 读取命令执行过程数据var reader = BufferedReader(InputStreamReader(process.inputStream))while (true) {val line = reader.readLine()if (line != null) {Log.i(TAG, "$line")}else{break}}reader.close()val exitValue = process.exitValue()Log.i(TAG, "压缩文件 , 执行完毕 , exitValue = $exitValue")}/*** 判定命令是否执行完毕* 调用 process.exitValue 方法 , 如果没有执行完毕 , 会抛异常,* 如果执行完毕会返回一个确定的值*/fun isComplete(process: Process): Boolean {try {// 已经执行完毕process.exitValue()return true} catch (e: IllegalThreadStateException) {// 未执行完毕return false}}/*** 使用 7zr 进行解压缩*/fun uncompress7z() {// /data/user/0/kim.hsl.a7_zip/files/7zrvar exeFile = File(filesDir, "7zr")// 执行前赋予可执行权限exeFile.setExecutable(true)// 删除解压目录var unzip_file = File("${filesDir.absolutePath}/unzip_file")if(unzip_file.exists()){recursionDeleteFile(unzip_file)}var cmd = "${exeFile.absolutePath} x ${filesDir.absolutePath}/files.7z -o${filesDir.absolutePath}/unzip_file"Log.i(TAG, "解压缩命令 : $cmd")var process: Process = Runtime.getRuntime().exec(cmd)// 读取命令执行过程数据var reader = BufferedReader(InputStreamReader(process.inputStream))while (true) {val line = reader.readLine()if (line != null) {Log.i(TAG, "$line")}else{break}}reader.close()val exitValue = process.exitValue()Log.i(TAG, "解压缩文件 , 执行完毕 , exitValue = $exitValue")}/*** 递归删除文件*/fun recursionDeleteFile(file: File) {if (file.isDirectory) {// 如果是目录 , 则递归删除file.listFiles().forEach {// ForEach 循环删除目录recursionDeleteFile(it)}} else {// 如果是文件直接删除file.delete()}}/*** 使用 7zr 进行压缩*/fun compress7zJni() {// /data/user/0/kim.hsl.a7_zip/files/7zrvar exeFile = File(filesDir, "7zr")// 执行前赋予可执行权限exeFile.setExecutable(true)// 删除原有的压缩文件, 如果存在var file_7z = File("${filesDir.absolutePath}/files_jni.7z")if(file_7z.exists()){file_7z.delete()}var cmd = "${exeFile.absolutePath} a ${filesDir.absolutePath}/files_jni.7z ${filesDir.absolutePath} -mx=9 -t7z"Log.i(TAG, "Jni 压缩命令 : $cmd")// 调用 jni 方法处理压缩文件executeCmd(cmd)}external fun executeCmd(cmd: String): Unit}

3、执行结果

-05-07 13:32:12.520 31022-31022/kim.hsl.a7_zip I/MainActivity: 开始拷贝 7zr 文件-05-07 13:32:12.524 31022-31022/kim.hsl.a7_zip I/MainActivity: filesDir = /data/user/0/kim.hsl.a7_zip/files , exeFile = /data/user/0/kim.hsl.a7_zip/files/7zr-05-07 13:32:12.525 31022-31022/kim.hsl.a7_zip I/MainActivity: 内置存储空间存在该 /data/user/0/kim.hsl.a7_zip/files/7zr 文件-05-07 13:32:12.527 31022-31022/kim.hsl.a7_zip I/MainActivity: 压缩命令 : /data/user/0/kim.hsl.a7_zip/files/7zr a /data/user/0/kim.hsl.a7_zip/files/files.7z /data/user/0/kim.hsl.a7_zip/files -mx=9 -t7z-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: 7-Zip (a) [64] 16.02 : Copyright (c) 1999- Igor Pavlov : -05-21-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: p7zip Version 16.02 (locale=utf8,Utf16=on,HugeFiles=on,64 bits,8 CPUs LE)-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Scanning the drive:-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: 11 folders, 8 files, 7403414 bytes (7230 KiB)-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Creating archive: /data/user/0/kim.hsl.a7_zip/files/files.7z-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Items to compress: 19-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Files read from disk: 8-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Archive size: 941905 bytes (920 KiB)-05-07 13:32:19.027 31022-31022/kim.hsl.a7_zip I/MainActivity: Everything is Ok-05-07 13:32:19.028 31022-31022/kim.hsl.a7_zip I/MainActivity: 压缩文件 , 执行完毕 , exitValue = 0-05-07 13:32:19.036 31022-31022/kim.hsl.a7_zip I/MainActivity: 解压缩命令 : /data/user/0/kim.hsl.a7_zip/files/7zr x /data/user/0/kim.hsl.a7_zip/files/files.7z -o/data/user/0/kim.hsl.a7_zip/files/unzip_file-05-07 13:32:19.059 31022-31022/kim.hsl.a7_zip I/MainActivity: 7-Zip (a) [64] 16.02 : Copyright (c) 1999- Igor Pavlov : -05-21-05-07 13:32:19.059 31022-31022/kim.hsl.a7_zip I/MainActivity: p7zip Version 16.02 (locale=utf8,Utf16=on,HugeFiles=on,64 bits,8 CPUs LE)-05-07 13:32:19.059 31022-31022/kim.hsl.a7_zip I/MainActivity: Scanning the drive for archives:-05-07 13:32:19.059 31022-31022/kim.hsl.a7_zip I/MainActivity: 1 file, 941905 bytes (920 KiB)-05-07 13:32:19.059 31022-31022/kim.hsl.a7_zip I/MainActivity: Extracting archive: /data/user/0/kim.hsl.a7_zip/files/files.7z-05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: ---05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: Path = /data/user/0/kim.hsl.a7_zip/files/files.7z-05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: Type = 7z-05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: Physical Size = 941905-05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: Headers Size = 333-05-07 13:32:19.185 31022-31022/kim.hsl.a7_zip I/MainActivity: Method = LZMA2:23-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Solid = +-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Blocks = 1-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Everything is Ok-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Folders: 11-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Files: 8-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Size: 7403414-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: Compressed: 941905-05-07 13:32:19.186 31022-31022/kim.hsl.a7_zip I/MainActivity: 解压缩文件 , 执行完毕 , exitValue = 0-05-07 13:32:19.187 31022-31022/kim.hsl.a7_zip I/MainActivity: Jni 压缩命令 : /data/user/0/kim.hsl.a7_zip/files/7zr a /data/user/0/kim.hsl.a7_zip/files/files_jni.7z /data/user/0/kim.hsl.a7_zip/files -mx=9 -t7z-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: jni 中处理压缩文件命令 : /data/user/0/kim.hsl.a7_zip/files/7zr a /data/user/0/kim.hsl.a7_zip/files/files_jni.7z /data/user/0/kim.hsl.a7_zip/files -mx=9 -t7z-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 0 . /data/user/0/kim.hsl.a7_zip/files/7zr-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 1 . a-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 2 . /data/user/0/kim.hsl.a7_zip/files/files_jni.7z-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 3 . /data/user/0/kim.hsl.a7_zip/files-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 4 . -mx=9-05-07 13:32:19.188 31022-31022/kim.hsl.a7_zip I/octopus: 5 . -t7z-05-07 13:32:26.301 31022-31022/kim.hsl.a7_zip I/octopus: 7zr 命令执行完毕 !

查询在 jni 层压缩的文件 :

C:\Users\octop>adb shellwalleye:/ $ suwalleye:/ # cd /data/user/0/kim.hsl.a7_zip/files/walleye:/data/user/0/kim.hsl.a7_zip/files # ls -latotal 3028drwxrwx--x 3 u0_a455 u0_a455 4096 -05-07 13:32 .drwx------ 5 u0_a455 u0_a455 4096 -05-07 13:19 ..-rwx------ 1 u0_a455 u0_a455 994304 -05-07 13:19 7zr-rw------- 1 u0_a455 u0_a455 941905 -05-07 13:32 files.7z-rw------- 1 u0_a455 u0_a455 1124135 -05-07 13:32 files_jni.7zdrwx------ 3 u0_a455 u0_a455 4096 -05-07 13:19 unzip_filewalleye:/data/user/0/kim.hsl.a7_zip/files #

四、参考资料

参考资料 :

7-Zip 官网 :https://www.7-/

Android NDK 编译构建脚本参考文档 :

ndk-build 脚本 :https://developer./ndk/guides/ndk-buildAndroid.mk 构建脚本: https://developer./ndk/guides/android_mkApplication.mk 构建脚本 :https://developer./ndk/guides/application_mk

博客资源 :源码 , 编译后的可执行文件, 在 7zip\p7zip_16.02\CPP\ANDROID\7zr\libs\ 目录下 ;

下载地址 :/download/han120/18215890GitHub 项目源码 :/han120/7-Zip

【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( jni 中 main 函数声明 | 命令行处理 | jni 调用 lib7zr.so 函数库处理压缩文件完整代码 )

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。