美文网首页
android ndk之libjpge.so 压缩图片

android ndk之libjpge.so 压缩图片

作者: 过期的薯条 | 来源:发表于2017-07-24 00:32 被阅读973次

    1.引言

    以前接触android的时候,压缩图片大多是用人家的,当时听到说qq用的是c压缩,觉得好搞大上。所以在自己学习ndk的时候,必须要写的例子就是用c来压缩图片。官方文档,界面丑了点,得到的信息还是很具有价值的。好过自己像无头苍蝇一样到处碰。

    2.正题

    2.1 准备jpeg库

    根据网上的博客,将库通过ndk-build编译得到我们需要的libjpeg.so文件。编译的博客地址Android_NDK图片压缩之Libjpeg库使用

    注意:
    git clone git://git.linaro.org/people/tomgall/libjpeg-turbo/libjpeg-turbo.git -b linaro-android 。假如 拉取失败换成https://git.linaro.org/people/tomgall/libjpeg-turbo/libjpeg-turbo.git -b linaro-android。

    2.2建立项目工程,导入头文件

    编译成功之后有俩个文件夹:obj,libs。其中obj存放的时头文件。libs存放的时so文件。将头文件倒入到cpp目录中这里我用的是jni目录。

    Paste_Image.png

    2.3导入成功之后编写Cmake

    引入第三方so的模版:

    
    #项目的文件路径,后续路径都是相对于这个路径G:\android-ndk-master\MyApplication\app\src\main\cpp\native-lib.cpp
    set(Project G:/android-ndk-master/MyApplication)
    #Cmake版本
    cmake_minimum_required(VERSION 3.4.1)
    
    #引入第三方库。根据文件路径查找
    add_library(jpeg  SHARED  IMPORTED)
    set_target_properties(jpeg PROPERTIES IMPORTED_LOCATION
                          ${Project}/app/libs/${ANDROID_ABI}/libjpeg.so)
    
    #引入第三方so的头文件
    include_directories( ${Project}/app/src/main/jni/jpeg )
    
    #引入自己编写的 c文件
    add_library(imagecompress  SHARED  ${Project}/app/src/main/jni/imagecompress.c)
    
    find_library(log-lib  log )
    
    #链接
    target_link_libraries(imagecompress  jpeg -ljnigraphics ${log-lib} )
    

    build gradle的配置:

    Paste_Image.png

    2.4 遇到的坑

    错误一:找不到自己写的 库”imagecompress “。首先检查Cmake等是否正确,然后看看activity中是否引用。第三生成的abi文件是否与当前的手机符合。
    本人出的错误就是:生产的abi:只有armeabi文件。。但是手机是x86的abi文件。。所以导致允许报链接错误。

    错误二

    Paste_Image.png

    这个错误很诡异: 提示无法关联到某一个方法。之所以诡异,是因为那些方法 你都可以在C代码中,通过Ctrl+右键 查看方法的头文件。并且自己也导入了头文件。原因:有些文件是ndk自带的,你可以正常的在C代码中引入。然后不报错,编译时报错的原因是,必须要在Cmake中链接本地ndk提供的so或者头文件。

    Paste_Image.png

    即使没引入。在写代码的时候 也不会报错,只有在编译的时候才会。这个时候需要在Cmake 添加链接库。

    Paste_Image.png

    关于jnigraphics 库的介绍博客:jnigraphics

    Paste_Image.png

    错误三
    cpp文件是c++的后缀。当在cpp文件中引入了c语言的方法。这个时候也会报错提示连接不上。只需要在方法前加上extern "C" ,这块很容易出错,所以 我索性全部都用c语言写。文件名也手动改成.c 而不是.cpp

    2.5 编写C语言代码啊:

    
    
    #include <android/bitmap.h>
    /*D:\android-ndk-r13b\platforms\android-13\arch-arm\usr\include\android\bitmap.h */
    
    
    #include <android/log.h>
    #include <stdio.h>
    #include <string.h>
    #include <setjmp.h>
    #include "jpeg/jpeglib.h"
    typedef uint8_t BYTE;
    #define TAG "image "
    #define LOGE(...) __android_log_print(ANDROID_LOG_INFO,TAG,__VA_ARGS__)
    
    #define true 1
    #define false 0
    const 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=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]);
       // LOGE("addon_message_table:%s", myerr->pub.addon_message_table);
    //  LOGE("SIZEOF:%d",myerr->pub.msg_parm.i[0]);
    //  LOGE("sizeof:%d",myerr->pub.msg_parm.i[1]);
       longjmp(myerr->setjmp_buffer, 1);
    }
    
    
    int generateJPEG(BYTE* data, int w, int h, int quality,
                    const char* outfilename, jboolean optimize) {
       int nComponent = 3;
    
       struct jpeg_compress_struct jcs;
    
       struct my_error_mgr jem;
    
       jcs.err = jpeg_std_error(&jem.pub);
       jem.pub.error_exit = my_error_exit;
       if (setjmp(jem.setjmp_buffer)) {
           return 0;
       }
       //为JPEG对象分配空间并初始化
       jpeg_create_compress(&jcs);
       //获取文件信息
       FILE* f = fopen(outfilename, "wb");
       if (f == NULL) {
           return 0;
       }
       //指定压缩数据源
       jpeg_stdio_dest(&jcs, f);
       jcs.image_width = w;
       jcs.image_height = h;
       if (optimize) {
           LOGE("optimize==ture");
       } else {
           LOGE("optimize==false");
       }
    
       jcs.arith_code = false;
       jcs.input_components = nComponent;
       if (nComponent == 1)
           jcs.in_color_space = JCS_GRAYSCALE;
       else
           jcs.in_color_space = JCS_RGB;
    
       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);
       }
    
       if (jcs.optimize_coding) {
           LOGE("optimize==ture");
       } else {
           LOGE("optimize==false");
       }
       //压缩完毕
       jpeg_finish_compress(&jcs);
       //释放资源
       jpeg_destroy_compress(&jcs);
       fclose(f);
    
       return 1;
    }
    
    
    char* jstringTostring(JNIEnv *env, jbyteArray barr) {
       char* rtn = NULL;
       jsize alen =(*env)->GetArrayLength(env,barr);
       jbyte * ba=(*env)->GetByteArrayElements(env,barr,0);
       if (alen > 0) {
           rtn = (char*) malloc(alen + 1);
           memcpy(rtn, ba, alen);
           rtn[alen] = 0;
       }
       (*env)->ReleaseByteArrayElements(env,barr,ba,0);
       return rtn;
    }
    
    
    
    JNIEXPORT jstring JNICALL
    Java_xinyi_com_myapplication_MainActivity_startCompress(JNIEnv *env, jclass jclass, jobject bitmap, jint width, jint height, jint quality, jbyteArray fileName, jboolean optimize) {
       AndroidBitmapInfo infoColor;
       int ret;
       BYTE *pixelColor;
       BYTE *data;
       BYTE *tempData;
       char *filename = jstringTostring(env,fileName);
       if((ret = AndroidBitmap_getInfo(env,bitmap,&infoColor)) < 0) {
           LOGE("解析错误111111");
           return (*env)->NewStringUTF(env,"0");
       }
    
    
       LOGE("解析错误222222");
       if(ret=AndroidBitmap_lockPixels(env,bitmap,(void**)&pixelColor)<0){
           LOGE("解析错误333333");
           return (*env)->NewStringUTF(env,"0");
       }
    
       BYTE r,g,b;
       int color;
       int w, h, format;
       w = infoColor.width;
       h = infoColor.height;
       format = infoColor.format;
    
       data = (BYTE *) malloc(infoColor.width * infoColor.height * 3);
    
       LOGE("解析错误444444");
    
       tempData = data;
       for(int i = 0; i < h; i++)
       {
           for(int j = 0; j < w; j++)
           {
              // LOGE("解析错误5555555");
               if (format == ANDROID_BITMAP_FORMAT_RGBA_8888) {
                   color = (*(int *) (pixelColor));
    
                   //LOGE("unsupported device: %d",color);
    
                   b = (color >> 16) & 0xFF;
                   g = (color >> 8) & 0xFF;
                   r = (color >> 0) & 0xFF;
                   *data = r;
                   *(data + 1) = g;
                   *(data + 2) = b;
    
                   data += 3;
                   pixelColor += 4;
    
               } else {
                   return -2;
               }
               //LOGE("解析错误666666");
           }
       }
       LOGE("解析错误777777");
    
       AndroidBitmap_unlockPixels(env,bitmap);
       int resultCode = generateJPEG(tempData,width,height,quality,filename,optimize);
       LOGE("解析错误888888");
       free(tempData);
       if(resultCode == 0) {
           jstring  result=(*env)->NewStringUTF(env,"0");
           return result;
       }
    
       return (*env)->NewStringUTF(env,"1");
    }
    

    其中为了让C语言输出的log 能显示到logcat中。
    实现Android Studio JNI开发C/C++使用__android_log_print输出Log

    注意:二重循环获取像素点不能在循环中进行io操作,否则的话会使运行时间大大加长。不加io操作,12000000次循环1秒钟完成加了一条io操作 起码得5.6分钟

    demo:https://github.com/wxy520ll/MyApplication

    相关文章

      网友评论

          本文标题:android ndk之libjpge.so 压缩图片

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