android平台C++代码约定

android不同于PC端,android标准输出(stdout)、标准错误(stderr)都被重定向到了/dev/null,printf ,std::cout函数根本不起作用。
android一般情况下,文件被放进assets目录下,该目录下所有的文件都会被打包至apk中。故model文件应放在此目录下。而放在该处的文件不能通过标准文件IO访问(没有正确的路径)。

约定1 使用封装的Log代码,而不是cout,printf。

Logging.h是封装后的代码,在CMakeLists指定编译宏SHOWLOG即可开启日志打印。若未定义宏SHOWLOG则日志不打印。

Logging.h

#pragma once

#ifdef __ANDROID__
#include <android/log.h>
#else
#include <iostream>
#include <stdio.h>
#endif

#ifdef __cplusplus
extern "C" {
#endif

#ifdef SHOWLOG
#ifdef __ANDROID__
#define LOGI(TAG,...)    __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__);
#define LOGW(TAG,...) __android_log_print(ANDROID_LOG_WARN, TAG, __VA_ARGS__);
#define LOGE(TAG,...)  __android_log_print(ANDROID_LOG_ERROR, TAG, __VA_ARGS__);
#define LOGD(TAG,...)  __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__);

#else
#define LOGI(TAG,...) \
     do { \
             std::cout<<__FILE__<<" "<<__LINE__<<"line"<<" "<<TAG<<":  "; \
             printf(__VA_ARGS__); } \
     while (0)

#define LOGW(TAG,...) \
     do {  \
             std::cout<<__FILE__<<" "<<__LINE__<<"line"<<" "<<TAG<<":  "; \
             printf(__VA_ARGS__); } \
     while (0)

#define LOGE(TAG,...) \
     do {  \
       std::cout<<__FILE__<<" "<<__LINE__<<"line"<<" "<<TAG<<":  "; \
       printf(__VA_ARGS__); } \
     while (0)
#define LOGD(TAG,...) \
     do {  \
       std::cout<<__FILE__<<" "<<__LINE__<<"line"<<" "<<TAG<<":  "; \
       printf(__VA_ARGS__); } \
     while (0)
#endif

#else
#define LOGI(TAG,...)
#define LOGW(TAG,...)
#define LOGE(TAG,...)
#define LOGD(TAG,...)
#endif

#ifdef __cplusplus
} /* extern "C" */
#endif

当需要打印日志时,类似于printfLOGD,LOGI,LOGW,LOGE有不同的日志等级

LOGD("TAG","log = %d",var1);

约定2 库代码尽量不直接从文件读取model,jni层把model读至内存,把内存地址以及长度传给sdk.

androidassets目录是无法通过文件地址访问的。android提供了<android/asset_manager_jni.h>来访问assets目录。为了避免model的多次copy,建议在jni层把model读至内存,前提是底层c++代码提供相关接口。

jni层代码

AAssetManager * manager = AAssetManager_fromJava(env,assetManager);//assetManager是java层对象
assert(NULL != manager);
std::string fileName = "model.zip";   //model名称
AAsset * asset = AAssetManager_open(manager, fileName.c_str(), AASSET_MODE_BUFFER);
assert(NULL != asset);
int size = AAsset_getLength(asset);
const char * buffer = static_cast<const char *>(AAsset_getBuffer(asset));

接口定义类似

void readModel(const char * buff, int bufLen);

约定3 model 压缩成model.gz.减少APK大小(对model可以压缩60%,格式是gz)。在内存中进行解压缩。

zlib是个跨平台库,android只需要在链接时,加上 -lz 即可。
解压缩函数如下:

#include <zlib.h>
#define CHUNK 100000
int decodeZip(const char *source,int len,int & length,char *dest)
{
    int ret;
    unsigned have;
    z_stream strm;
    unsigned char out[CHUNK];
    int totalsize = 0;

    /* allocate inflate state */
    strm.zalloc = Z_NULL;
    strm.zfree = Z_NULL;
    strm.opaque = Z_NULL;
    strm.avail_in = 0;
    strm.next_in = Z_NULL;

    ret = inflateInit2(&strm, 31);

    if (ret != Z_OK)
        return ret;

    strm.avail_in = len;
    strm.next_in = (unsigned char*)source;

    /* run inflate() on input until output buffer not full */
    do {
        strm.avail_out = CHUNK;
        strm.next_out = out;
        ret = inflate(&strm, Z_NO_FLUSH);
        switch (ret)
        {
            case Z_NEED_DICT:
                ret = Z_DATA_ERROR; /* and fall through */
            case Z_DATA_ERROR:
            case Z_MEM_ERROR:
                inflateEnd(&strm);
                return ret;
        }

        have = CHUNK - strm.avail_out;
        memcpy(dest + totalsize ,out,have);
        totalsize += have;

    } while (strm.avail_out == 0);

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,082评论 25 707
  • 1:InputChannel提供函数创建底层的Pipe对象 2: 1)客户端需要新建窗口 2)new ViewRo...
    自由人是工程师阅读 5,301评论 0 18
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,402评论 0 17
  • 《腾讯桌球:客户端总结》 本次分享总结,起源于腾讯桌球项目,但是不仅仅限于项目本身。虽然基于Unity3D,很多东...
    吴秦阅读 24,451评论 12 142
  • 人没有必要与世界格格不入,也没必要让自己蝇营狗苟,违背自己的良知而迎合错误的东西。接受与自己契合的事物,把握能够改...
    爱学习的我阅读 221评论 1 3