android - 从字节数组 (byte[]) 转换位图 - 位图仍然为空

标签 android bitmap java-native-interface arrays decode

我从应用程序的 JNI 部分检索了一个字节数组。 JNI 的输入也是来自相机的 ARGB8888 格式位图图片的字节数组。 C++ 使用 FastCV 函数将我的位图修改为灰度。这是我的代码:

Java:

private void fcvTest(String fileName) { //path to stored image 

        Bitmap bmp = BitmapFactory.decodeFile(fileName);
        //we are going from 32bit integer (ARGB8888) to 8bit byte (grayscale) => 4 * number of pixels:
        ByteBuffer argbBuf = ByteBuffer.allocate(bmp.getWidth() * bmp.getHeight() * 4); 
        bmp.copyPixelsToBuffer(argbBuf);
         byte[] grayBuffer = new byte[]{};
        grayBuffer = fcvGrayScale(argbBuf.array(), bmp.getWidth(), bmp.getHeight());


        Bitmap grayScaledPic = BitmapFactory.decodeByteArray(grayBuffer, 0, grayBuffer.length);

    }

private native byte[] fcvGrayScale(byte[] data, int w, int h);

C++:

JNIEXPORT jbyteArray JNICALL Java_com_qualcomm_loadjpeg_LoadJpeg_fcvGrayScale(JNIEnv* env, jobject obj, jbyteArray img, uint32_t w, uint32_t h) {

    jbyte* jImg;
    jboolean isCopy;
    jbyteArray grayToJava;
    jbyte* jbyteGray;

    jImg = env->GetByteArrayElements(img, &isCopy);
    uint8_t* uint8Gray = (uint8_t*) fcvMemAlloc(w*h*4, 16);

    fcvColorRGB888ToGrayu8((uint8_t*) jImg, w, h, 0, uint8Gray, 0);

    jbyteGray = (jbyte*) uint8Gray;
    grayToJava = env->NewByteArray(w*h); //no need for 4-more space for grayscale
    env->SetByteArrayRegion(grayToJava, 0, w*h, jbyteGray);

    env->ReleaseByteArrayElements(img, jImg, JNI_ABORT);
    fcvMemFree(uint8Gray);

    return grayToJava;
}

可以在此处查看 FastCV 函数 fcvColorRGB888ToGrayu8 的文档: fcvColorRGB888ToGrayu8 doc

正如您在标题中看到的,我无法从 JNI 设置新位图 grayScaledPic。 Log cat 给了我这个通常的声明:

SkImageDecoder::Factory returned null

我的新数组 grayBuffer 中的字节似乎不正确,或者我忽略了一些问题,例如通过 BitmapFactory.Options 设置正确的位图格式。或者输入数组甚至可能是错误的,因为 fcvColorRGB888ToGrayu8 期望 RGB888 作为输入,但 Android 不支持这种格式。我必须先删除 alpha channel 吗?

有谁知道,还有什么意义?

已解决 >> 查看已接受的答案

问题出在通过 BitmapFactory.decodeByteArray 解码字节数组时 - 仅适用于压缩位图(JPEG、PNG)

最佳答案

问题出在您的 Java 代码中:

Bitmap grayScaledPic = BitmapFactory.decodeByteArray(grayBuffer, 0, grayBuffer.length);

BitmapFactor.decodeByteArray()解码已加载到字节数组中的压缩图像,例如 JPG 或 PNG 文件。它不适用于未压缩的原始数据。

要从灰度字节数组进行设置,您必须使用不同的方法。您可以转换为 int 数组,然后使用 Bitmap.createBitmap() 创建位图例如:

final int pixCount = bmp.getWidth() * bmp.getHeight();
int[] intGreyBuffer = new int[pixCount];
for(int i=0; i < pixCount; i++)
{
    int greyValue = (int)greyBuffer[i] & 0xff;
    intGreyBuffer[i] = 0xff000000 | (greyValue << 16) | (greyValue << 8) | greyValue;
}
Bitmap grayScaledPic = Bitmap.createBitmap(intGreyBuffer, bmp.getWidth(), bmp.getHeight(), Bitmap.Config.ARGB_8888);

另一种方法是从 JNI 代码返回一个 int 数组而不是字节数组,然后直接在 Java 中从中创建位图。

关于android - 从字节数组 (byte[]) 转换位图 - 位图仍然为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28334195/

相关文章:

android - 阻止 Android ContentObserver 被多次注册

android - 如何在android IconGenerator上将文本居中

Android - 加载位图而不改变

java - Java 和 C 是否以相同的方式表示 UTF char 字节?

java - 如何定义 swig typemap 以将 unsigned char* 返回给 java

java - 查找与 DLL 中的偏移量匹配的 Delphi 源代码行

android - 如何在我的申请中获得 GPRMC 的 NMEA 判决?

android - 根据另一个微调器的选定项目更新微调器

android - 在 android 的 ScrollView 中加载数百个位图图像时出现内存不足错误

android - 通过给定的 GPS 坐标跟踪车辆应用程序