0°

Android图片压缩加密上传 – NDK终极压缩和加密上传

文章内容大纲 (右击展开→→)

内容预览:
  • 值得爱学习的你去关注,感觉有帮助转发分享让更多的人去关注!点击上方...~
  •   我们将已经编译好的libjpeg.so以及.h头文件拷贝到jni目录下面,就可...~
  • 对于老的通过Android.mk文件编译的NDK项目,直接一条配置整个项目就可以...~

始发于微信公众号: 程序员小乐

Android图片压缩加密上传 - NDK终极压缩和加密上传


分享互联网、技术、情感等干货平台。您还可以向公众号投稿,将自己总结的技术、心得、经验分享给大家。学无止境,不求尽如人意,但求问心无愧。让学习成为一种美、一种习惯。值得爱学习的你去关注,感觉有帮助转发分享让更多的人去关注!点击上方蓝字关注!

作者:红橙Darren

链接:http://www.jianshu.com/p/eebe2107da6d

这段时间比较忙发干货也少了,出现取消关注了,我会继续为大家更新干货的,谢谢大家的关注和批评。

1. 概述

上一期已讲到,我们不打算采用BitmapFactory去压缩,而是采用JPEG的压缩算法,当然大家最好是将两者结合一下,今天我们直接去网上找一个已经写好的开源库,然后我们在他的基础上再写一些Native代码就好,当然也可以自己一步一步去写算法处理。

Android图片压缩加密上传 - NDK终极压缩和加密上传

效果演示


所有分享大纲:(http://www.jianshu.com/p/c0ec2a7fc26a)

视频讲解地址:关注公众号发送“NDK”获得视频

2. 编译libjpeg.so库文件


  关于C和C++以及NDK的基础我这里就不强调,如果大家觉得我写的C++代码看不懂那么也没关系,你只需要关注我这里所实现的思路,我最终把它编译成.so库你能用就行。
  打开
 下载已经提供好的开源库,打开后你会看到有很多的.c文件和.h头文件,把他编译成.so库文件即可,当然道路曲折会出现很多问题,这里我就不写过程了,等到增量更新的时候我会一步一步去带大家编译的。
  我们将已经编译好的libjpeg.so以及.h头文件拷贝到jni目录下面,就可以开始写压缩代码了:

Android图片压缩加密上传 - NDK终极压缩和加密上传

目前jni目录

3. 编写imgcompcrypt.cpp


  我使用的是AS,网上关于AS的NDK方面的资料比较少,需要多花一些功夫,最好把AS升级到2.2版本以上,然后下载NDK、CMake、LLDB,一个支持、一个插件、一个调试:

Android图片压缩加密上传 - NDK终极压缩和加密上传

{4U3J{YTF`%)}1T4)W6LUH1.png

  升级到2.2之后我们写C和C++的代码才会有提示,之前的版本我还没找到解决的方案,网上搜索了很多也没有相关答案,而且它也支持Cmake和ndk-build两种方式,相比与以前的gradle去配置ndk编译目录什么的简直是方便多了。对于老的通过Android.mk文件编译的NDK项目,直接一条配置整个项目就可以被AS支持了。如果觉得Cmake的方式不习惯还是可以采用ndk-build的方式这点倒是无所谓,至于代码提示肯定是要的这个很致命,要不然写代码会比较慢。怎么生成头文件我就不讲了,这里直接上代码:

#include "imgcompcrypt.h"
#include <string.h>
#include <android/bitmap.h>
#include <android/log.h>
#include <stdio.h>
#include <setjmp.h>
#include <math.h>
#include <stdint.h>
#include <time.h>
//统一编译方式
extern "C" {
#include "jpeg/jpeglib.h"
#include "jpeg/cdjpeg.h"        
/* Common decls for cjpeg/djpeg applications */
#include "jpeg/jversion.h"        
/* for version message */
#include "jpeg/jconfig.h"#include "filecrypt.c"
}
// log打印#define LOG_TAG "jni"
#define LOGW(...)  __android_log_write(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
#define true 1#define false 0

typedef uint8_t BYTE;
// error 结构体
char *error;
struct my_error_mgr {    
struct jpeg_error_mgr pub;    jmp_buf setjmp_buffer; };
   typedef struct my_error_mgr *my_error_ptr;    METHODDEF(void)    my_error_exit(j_common_ptr cinfo) {    my_error_ptr myerr = (my_error_ptr) cinfo->err;    (*cinfo->err->output_message)(cinfo);    error = (char *) myerr->pub.jpeg_message_table[myerr->pub.msg_code];    LOGE("jpeg_message_table[%d]:%s", myerr->pub.msg_code,         myerr->pub.jpeg_message_table[myerr->pub.msg_code]);    longjmp(myerr->setjmp_buffer, 1); }
   int generateJPEG(BYTE *data, int w, int h, int quality,  const char *outfilename, jboolean optimize) {    
   // 结构体相当于Java类    struct jpeg_compress_struct jcs;  
   //当读完整个文件的时候就会回调my_error_exit这个退出方法。    struct my_error_mgr jem;    jcs.err = jpeg_std_error(&jem.pub);    jem.pub.error_exit = my_error_exit;  
   // setjmp是一个系统级函数,是一个回调。    if (setjmp(jem.setjmp_buffer)) {        
       return 0;    }    //初始化jsc结构体    jpeg_create_compress(&jcs);    
   //打开输出文件 wb 可写  rb 可读    FILE *f = fopen(outfilename, "wb");    
   if (f == NULL) {        
       return 0;    }    
   //设置结构体的文件路径,以及宽高    jpeg_stdio_dest(&jcs, f);    jcs.image_width = w;    jcs.image_height = h;  
   //
   /* TRUE=arithmetic coding, FALSE=Huffman
   */
   jcs.arith_code = false;    
   int nComponent = 3;    
   /* 颜色的组成 rgb,三个
   # of color components in input image */
   jcs.input_components = nComponent;    
   //设置颜色空间为rgb    jcs.in_color_space = JCS_RGB;    
   ///* Default parameter setup for compression */    jpeg_set_defaults(&jcs);    
   //是否采用哈弗曼    jcs.optimize_coding = optimize;    
   //设置质量    jpeg_set_quality(&jcs, quality, true);    
   //开始压缩    jpeg_start_compress(&jcs, TRUE);    JSAMPROW row_pointer[1];    
   int row_stride;    row_stride = jcs.image_width * nComponent;    
   while (jcs.next_scanline < jcs.image_height) {        
   //得到一行的首地址    row_pointer[0] = &data[jcs.next_scanline * row_stride];    jpeg_write_scanlines(&jcs, row_pointer, 1);    }    
   // 压缩结束    jpeg_finish_compress(&jcs);    
   // 销毁回收内存    jpeg_destroy_compress(&jcs);    
   //关闭文件    fclose(f);    
     return 1;    }
   jint Java_net_bither_util_NativeUtil_compressBitmap(JNIEnv *env,    jclass thiz, jobject bitmap, int quality,    jstring fileNameStr, jboolean optimize) {    
   // 1.获取Bitmap信息    AndroidBitmapInfo android_bitmap_info;    AndroidBitmap_getInfo(env, bitmap, &android_bitmap_info);    
   // 获取bitmap的 宽,高,format    int bitmap_width = android_bitmap_info.width;    
   int bitmap_height = android_bitmap_info.height;    
   int format = android_bitmap_info.format;    
   if (format != ANDROID_BITMAP_FORMAT_RGBA_8888) {        
       return -1;    }    
   // 2.解析Bitmap的像素信息,并转换成RGB数据,保存到二维byte数组里面    BYTE *pixelscolor;    
   // 2.1 锁定画布    AndroidBitmap_lockPixels(env, bitmap, (void **) &pixelscolor);    
   // 2.2 解析初始化参数值    BYTE *data;    BYTE r, g, b;    data = (BYTE *) malloc(bitmap_width * bitmap_height * 3);
   //每一个像素都有三个信息RGB    BYTE *tmpData;    tmpData = data;
   //临时保存data的首地址    int i = 0, j = 0;    
   int color;    
   //2.3 解析每一个像素点里面的rgb值(去掉alpha值),保存到一维数组data里面    for (i = 0; i < bitmap_height; ++i) {        
       for (j = 0; j < bitmap_width; ++j) {            
           //获取二维数组的每一个像素信息首地址            color = *((int *) pixelscolor);            r = ((color & 0x00FF0000) >> 16);            g = ((color & 0x0000FF00) >> 8);            b = ((color & 0x000000FF));            
           //保存到data数据里面            *data = b;            *(data + 1) = g;            *(data + 2) = r;            data = data + 3;            
           // 一个像素包括argb四个值,每+4就是取下一个像素点            pixelscolor += 4;        }    }    
   // 2.4. 解锁Bitmap    AndroidBitmap_unlockPixels(env, bitmap);    
   // jstring --> c char    char *fileName = (char*)(env)->GetStringUTFChars(fileNameStr, 0);    
   //3. 调用libjpeg核心方法实现压缩    int resultCode = generateJPEG(tmpData, bitmap_width, bitmap_height, quality, fileName, optimize);    
   //4.释放资源    env->ReleaseStringUTFChars(fileNameStr, fileName);    
      free((void *) tmpData);    
    // 4.2 释放Bitmap     // 4.2.1 通过对象获取类    jclass bitmap_clz = env->GetObjectClass(bitmap);    
   // 4.2.2 通过类和方法签名获取方法id    jmethodID recycle_mid = env->GetMethodID(bitmap_clz, "recycle", "()V");    
   // 4.2.3 执行回收释放方法    env->CallVoidMethod(bitmap, recycle_mid);    
   // 5.返回结果    if (resultCode == 0) {        
       return -1;    }    
    return 1; }

3. 图片文件加密


  文件的加密相对来说就比较简单了,因为可能可很多地方涉及到文件加密,这里我就把文件加密单独分开了,当然也可以写到图片压缩一起,可以用C写也可以用C++因为上面是用的C++,那么加密我们就采用C:

    #include <jni.h>
   #include <stdlib.h>
   #include <stdio.h>
   #include <string.h>
   #include "net_bither_util_FileCrypt.h"

   // 加密的秘钥char password[] = "Big god take me fly!";
   // 加密文件void crypt_file(char *normal_path, char *crypt_path) {    
   // 打开文件    FILE *normal_fp = fopen(normal_path, "rb");    FILE *crypt_fp = fopen(crypt_path, "wb");    
   //一次读取一个字符    int ch;    
   int i = 0;
   //循环使用密码中的字母进行异或运算    int pwd_len = strlen(password);
   //密码的长度    while ((ch = fgetc(normal_fp)) != EOF) {
   //End of File    //写入(异或运算)    fputc(ch ^ password[i % pwd_len], crypt_fp);      i++;    }    
   // 关闭    fclose(crypt_fp);    fclose(normal_fp); }
   // 加密文件,jfile_path 源文件路径  jcrypt_path 加密后文件路径
   JNIEXPORT void JNICALL Java_com_hc_filecrypt_FileCrypt_cryptFile    (JNIEnv *env, jclass jclazz, jstring jfile_path, jstring jcrypt_path) {    
   char *normal_path = (char*)(*env)->GetStringUTFChars(env, jfile_path, JNI_FALSE);    
   char *crypt_path = (char*)(*env)->GetStringUTFChars(env, jcrypt_path, JNI_FALSE);    crypt_file(normal_path, crypt_path); }

4. 最后的测试


  接近3M的原图压缩到30K,可以找找哪一张是被压缩过的,会比我们使用BitmapFarctory或者Bitmap.compress()压缩出来的一些效果要好很多:

Android图片压缩加密上传 - NDK终极压缩和加密上传

效果对比

所有分享大纲:

视频讲解地址:关注公众号发送“NDK”获得视频

如果您觉得不错,请别忘了分享到您的朋友圈让更多的人看到!! 您的举手之劳,就是对我最好的支持,非常感谢!

推荐文章(关注公众号查看往期文章):
学习资料(干货汇集)不断更新【更新于2017-2-25】
2017 年初、阿里、腾讯、百度、华为、京东、搜狗和滴滴面试题汇集(更新篇)
如果你喜欢上了一个程序员小伙,献给所有的程序员女友

大家可以加群和大牛们一起学习,由于“大牛聚集之地、大牛聚集之地2群”已满
如果想进群,可以加我的微信,我拉你进群,我的微信:1733563441(请备注)

看完本文有收获?请转发分享给更多人
关注「杨守乐」,提升编程技能

Android图片压缩加密上传 - NDK终极压缩和加密上传

如何关注:

① 长按二维码,选择“识别图中二维码”进行关注。

 点击微信右上角的“+”,会出现“添加朋友”,进入“公众号”,输入公众号的 “杨守乐”名字,即可找到。

如果你有好的文章想和大家分享,欢迎投稿,直接向我投递文章链接即可。
投稿邮箱:[email protected]/[email protected]
欢迎扫描关注我们的微信公众号“杨守乐”,不要错过每一篇干货~

技术社区:
http://blog.csdn.net/xiaole0313

【QQ技术群】279126311 [满]
【QQ技术群】484572225 [未]

以上就是:Android图片压缩加密上传 - NDK终极压缩和加密上传 的全部内容。

本站部分内容来源于互联网和用户投稿,如有侵权请联系我们删除,谢谢。
Email:[email protected]


0 条回复 A 作者 M 管理员
    所有的伟大,都源于一个勇敢的开始!
欢迎您,新朋友,感谢参与互动!欢迎您 {{author}},您在本站有{{commentsCount}}条评论