我正在使用相机中的previewCallback尝试抓取图像。这是我正在使用的代码
private Camera.PreviewCallback mPrevCallback = new Camera.PreviewCallback()
{
public void onPreviewFrame( byte[] data, Camera Cam ) {
Log.d("CombineTestActivity", "Preview started");
Log.d("CombineTestActivity", "Data length = "
+ data.length );
currentprev = BitmapFactory.decodeByteArray( data, 0,
data.length );
if( currentprev == null )
Log.d("CombineTestActivity", "currentprev is null" );
Log.d("CombineTestActivity", "Preview Finished" );
}
};
数据的长度总是与576000相同。
此外,我尝试更改相机的参数,以便图像以不同的格式返回。这就是我这样做的样子。
mCamera = Camera.open();
camParam = mCamera.getParameters();
camParam.setPreviewFormat( ImageFormat.RGB_565 );
mCamera.setParameters( camParam );
mCamera.setPreviewCallback( mPrevCallback );
然而,当我更改预览格式并将其保留为NV21的默认格式时,BitmapFactory.decodeByteArray将返回null。我也尝试将预览格式更改为JPEG类型。我甚至在ddms中得到了一个调试语句,这就是我得到的
“D / skia(14391):--- SkImageDecoder :: Factory返回null”
答案 0 :(得分:41)
好的,希望这会有所帮助。
搜索互联网寻找快速解决方案,并找到了完美的东西。
适用于Android 2.1
感谢来自this page的off3nsiv3。
// Convert to JPG
Size previewSize = camera.getParameters().getPreviewSize();
YuvImage yuvimage=new YuvImage(data, ImageFormat.NV21, previewSize.width, previewSize.height, null);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
yuvimage.compressToJpeg(new Rect(0, 0, previewSize.width, previewSize.height), 80, baos);
byte[] jdata = baos.toByteArray();
// Convert to Bitmap
Bitmap bmp = BitmapFactory.decodeByteArray(jdata, 0, jdata.length);
对off3nsiv3的代码稍作修改就可以了。与手动解码相比,FPS仍然非常高。
对于上面的代码,80是jpeg质量(0表示100,100表示最佳)。
答案 1 :(得分:7)
你可以尝试这个并且它正在工作......
mCamera.setOneShotPreviewCallback(new Camera.PreviewCallback() {
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
Camera.Parameters parameters = camera.getParameters();
int format = parameters.getPreviewFormat();
//YUV formats require more conversion
if (format == ImageFormat.NV21 || format == ImageFormat.YUY2 || format == ImageFormat.NV16) {
int w = parameters.getPreviewSize().width;
int h = parameters.getPreviewSize().height;
// Get the YuV image
YuvImage yuv_image = new YuvImage(data, format, w, h, null);
// Convert YuV to Jpeg
Rect rect = new Rect(0, 0, w, h);
ByteArrayOutputStream output_stream = new ByteArrayOutputStream();
yuv_image.compressToJpeg(rect, 100, output_stream);
byte[] byt = output_stream.toByteArray();
FileOutputStream outStream = null;
try {
// Write to SD Card
File file = createFileInSDCard(FOLDER_PATH, "Image_"+System.currentTimeMillis()+".jpg");
//Uri uriSavedImage = Uri.fromFile(file);
outStream = new FileOutputStream(file);
outStream.write(byt);
outStream.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
}
}
}
答案 2 :(得分:3)
我正在尝试做同样的事情。基于讨论here和here,听起来人们从Android 2.1 / 2.2开始就没有幸运decodeByteArray()
来处理NV21格式。它绝对不适用于我的模拟器或我的Droid Incredible,虽然我认为这会调用本机代码,因此它可能会在某些手机上运行,具体取决于驱动程序?
作为替代方案,您可以尝试使用Java自己解码NV21(请参阅上面的链接以获取示例),尽管这对于大多数情况来说显然太慢而无法使用。我没有太多运气试图让CameraPreview
发送不同的格式,我希望这对于尝试编写可在不同硬件上移植的代码是有问题的。如果你在NDK中编写了NV21解码方法,你可能会稍微提高帧速率。
显然,在尝试处理CameraPreview
时,由于竞争条件存在稳定性问题,尽管我自己没有证实这个问题。我认为您可以通过使用Android 2.1 / 2.2中添加的缓冲预览回调方法setPreviewCallbackWithBuffer()
来避免这种情况并提高帧速率。这是在2.1中添加的,但保持隐藏直到2.2。 To use it in 2.1 you need to hack around the hiding.
有些人建议使用MediaRecorder
代替CameraPreview
。不幸的是,MediaRecorder
似乎提供的预览图片比CameraPreview
更少,所以我不推荐这条路线。
答案 3 :(得分:1)
从我之前的回答中更新,但请注意Qix的答案,看起来更简单
我实际上在纯Java中解码效果不错。只要预览大小不是太大,帧速率大约为10-15 fps。可悲的是,我的Droid公司的Android 2.3更新似乎已经取消了一些较小的预览尺寸选项:(。我也试过在我从另一个项目中提取的本机代码中执行此操作,但这是错误的,似乎没有任何更快对于我正在进行的相对简单的处理,所以我没有进一步追求它。请参阅my Github获取源代码(包括Java和本机)。
答案 4 :(得分:0)
这对我有用...
private Camera.PreviewCallback getPreviewCallback() {
Log.d("TAG", "previewCallBack ==> " + "getPreviewCallback");
Camera.PreviewCallback previewBitmap = new Camera.PreviewCallback() {
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
Camera.Size previewSize = camera.getParameters().getPreviewSize();
YuvImage yuvimage=new YuvImage(data, ImageFormat.NV21, previewSize.width, previewSize.height, null);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
yuvimage.compressToJpeg(new Rect(0, 0, previewSize.width, previewSize.height), 80, baos);
byte[] jdata = baos.toByteArray();
// Convert to Bitmap
Bitmap bitmap = BitmapFactory.decodeByteArray(jdata, 0, jdata.length);
Log.d("TAG", "BITMAP ==> " + bitmap);
runModelInference(bitmap);
}
};
return previewBitmap;
}
答案 5 :(得分:-1)
尝试如下:
public Bitmap stringtoBitmap(String string) {
Bitmap bitmap = null;
try {
YuvImage yuvimage = new YuvImage(base64.getBytes(),ImageFormat.YUY2, 120, 30, null);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
yuvimage.compressToJpeg(new Rect(0, 0, 20, 20), 100, baos);
byte[] jdata = baos.toByteArray();
bitmap = BitmapFactory.decodeByteArray(jdata, 0, jdata.length);
} catch (Exception e) {
}
return bitmap;
}