我正在测试使用 OpenCv 在 JNI 层上操作位图像素的性能。
两个选项:
1. 传递整数数组,操作像素并将像素写回位图。[41ms]
2. 传递整个 Bitmap。[35毫秒]
我注意到传递 Bitmap 比传递像素数组并将数组分配给 Bitmap 快大约 5 毫秒。
问题是使用选项 2 我失去了蓝色并得到橙色代替我期望的蓝色。图像是 ARGB 并且似乎已更改为 RGBA?可能是什么问题?
方法一:
Java
Bitmap out = Bitmap.createBitmap(width, height, Config.ARGB_8888);
int[] rgba = new int[width*height];
mSmasher.loadImage(imagePath, rgba, 0);
out.setPixels(rgba, 0, width, 0, 0, width, height);
JNI
JNIEXPORT void JNICALL Java_com_vblast_smasher_Smasher_loadImage
(JNIEnv *pEnv, jobject obj, jstring jFilePath, jintArray jbgra, jint options)
{
jint* _bgra = pEnv->GetIntArrayElements(jbgra, 0);
const char *filePath = pEnv->GetStringUTFChars(jFilePath, 0);
if (NULL != filePath)
{
// init our output image
Mat bgra(outputHeight, outputWidth, CV_8UC4, (unsigned char *)_bgra);
// bgra image manipulations
}
pEnv->ReleaseIntArrayElements(jbgra, _bgra, 0);
pEnv->ReleaseStringUTFChars(jFilePath, filePath);
}
方法二:
JNIEXPORT void JNICALL Java_com_vblast_smasher_Smasher_getLayersBitmap
(JNIEnv *pEnv, jobject obj, jobject bitmap)
{
int ret;
AndroidBitmapInfo info;
void* pixels = 0;
if ((ret = AndroidBitmap_getInfo(pEnv, bitmap, &info)) < 0) {
LOGE("AndroidBitmap_getInfo() failed ! error=%d", ret);
return;
}
if (info.format != ANDROID_BITMAP_FORMAT_RGBA_8888 )
{
LOGE("Bitmap format is not RGBA_8888!");
return;
}
if ((ret = AndroidBitmap_lockPixels(pEnv, bitmap, &pixels)) < 0) {
LOGE("AndroidBitmap_lockPixels() failed ! error=%d", ret);
}
// init our output image
Mat mbgra(info.height, info.width, CV_8UC4, pixels);
mLayers[0].copyTo(mbgra);
AndroidBitmap_unlockPixels(pEnv, bitmap);
}
解决方案 05/23:
问题是当使用整数数组并将其传递给 JNI 时,字节顺序从 ARGB (java) 更改为 BGRA(native)。这对于处理像素很好。但是,在锁定像素时传递实际的位图对象,字节顺序没有改变,因此需要在修改像素数据后进行更改。
cvtColor(mbgra, mbgra, COLOR_BGR2RGBA, 4);