6

我想使用 JNI 将位图从 Android 传递到 C++ 以进行 Android 开发。

在 Java 中,我调用此函数将位图从 Java 发送到 C++:

public native int sendMyBitmap(Bitmap bitmap);

在 JNI 中,我是这样做的:

JNIEXPORT void JNICALL sendMyBitmap(JNIEnv * env,
        jobject obj, jobject bitmap) 
{
    AndroidBitmapInfo androidBitmapInfo ;
    void* pixels;
    AndroidBitmap_getInfo(env, bitmap, &androidBitmapInfo);
    AndroidBitmap_lockPixels(env, bitmap, &pixels);
    unsigned char* pixelsChar = (unsigned char*) pixels;
    saveImage(pixelsChar);
}

void saveImage(unsigned char* img)
{
    FILE *f;
    int w = 640, h = 480;

    int filesize = 54 + 3*w*h;  //w is your image width, h is image height, both int

    unsigned char bmpfileheader[14] = {'B','M', 0,0,0,0, 0,0, 0,0, 54,0,0,0};
    unsigned char bmpinfoheader[40] = {40,0,0,0, 0,0,0,0, 0,0,0,0, 1,0, 24,0};
    unsigned char bmppad[3] = {0,0,0};

    bmpfileheader[ 2] = (unsigned char)(filesize    );
    bmpfileheader[ 3] = (unsigned char)(filesize>> 8);
    bmpfileheader[ 4] = (unsigned char)(filesize>>16);
    bmpfileheader[ 5] = (unsigned char)(filesize>>24);

    bmpinfoheader[ 4] = (unsigned char)(       w    );
    bmpinfoheader[ 5] = (unsigned char)(       w>> 8);
    bmpinfoheader[ 6] = (unsigned char)(       w>>16);
    bmpinfoheader[ 7] = (unsigned char)(       w>>24);
    bmpinfoheader[ 8] = (unsigned char)(       h    );
    bmpinfoheader[ 9] = (unsigned char)(       h>> 8);
    bmpinfoheader[10] = (unsigned char)(       h>>16);
    bmpinfoheader[11] = (unsigned char)(       h>>24);

    f = fopen("/storage/test.png","wb");
    fwrite(bmpfileheader,1,14,f);
    fwrite(bmpinfoheader,1,40,f);
    for(int i=0; i<h; i++)
    {
        fwrite(img+(w*(h-i-1)*3),3,w,f);
        fwrite(bmppad,1,(4-(w*3)%4)%4,f);
    }
    fflush(f);
    fclose(f);
}

在 C++ JNI 中保存位图后,Java 和 C++ 中的位图不同。我正在努力找出问题所在以及如何找到问题所在。

4

1 回答 1

3

这一行:

fwrite(img + (w * (h - i - 1)*3), 3, w, f);

您假设输入 BITMAP 每个像素有 24 位(3 个字节)。但是android中没有这样的像素格式。

您还假设宽度正好是 640 像素,高度正好是 480 像素。在您的情况下这可能是事实,但您应该阅读此信息,而不是对值进行硬编码。

所有这些值都由函数提供AndroidBitmap_getInfo()。你为什么不使用它们?只需检查结构AndroidBitmapInfo

typedef struct {
    uint32_t    width;
    uint32_t    height;
    uint32_t    stride;
    int32_t     format;
    uint32_t    flags;      // 0 for now
} AndroidBitmapInfo;

您可以直接读取位图的尺寸。此外,您应该检查参数format,它可以是以下值之一:

enum AndroidBitmapFormat {
    ANDROID_BITMAP_FORMAT_NONE      = 0,
    ANDROID_BITMAP_FORMAT_RGBA_8888 = 1,
    ANDROID_BITMAP_FORMAT_RGB_565   = 4,
    ANDROID_BITMAP_FORMAT_RGBA_4444 = 7,
    ANDROID_BITMAP_FORMAT_A_8       = 8,
};

你看?没有RGB_888格式。您应该确保RGBA_8888从 Java co C++ 发送像素格式的位图并相应地修改您的算法。我认为将输出像素格式更改为 32 位(而不是 24 位)会更容易。

于 2014-03-13T16:17:53.493 回答