我有一些拍照的代码在Android 2.1和2.2中有效。 但这些代码在Android 2.3上爆发。 花了一些时间来解决这个徒劳的问题后,我想在这里寻求帮助。
我拍摄照片的代码流如下:
创建一个类Camlayer扩展SurfaceView
public class CamLayer extends SurfaceView implements SurfaceHolder.Callback {
private void init(Context context){
// Install a SurfaceHolder.Callback so we get notified when the
// underlying surface is created and destroyed.
mHolder = getHolder();
mHolder.addCallback(this);
mHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
mCamera = Camera.open();
}
public CamLayer(Context context) {
super(context);
init(context);
}
public CamLayer(Context context, AttributeSet attrs) {
super(context, attrs);
init(context);
}
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
Log.i(TAG+".surfaceChanged", "being called!");
Log.i(TAG+".surfaceChanged", "w="+w);
Log.i(TAG+".surfaceChanged", "h="+h);
if (isPreviewRunning) {
mCamera.stopPreview();
}
try {
mCamera.setPreviewDisplay(holder);
mCamera.setPreviewCallback(mPreviewCallback);
} catch (IOException e) {
Log.e(TAG+".surfaceCreated", "mCamera.setPreviewDisplay(holder);");
}
Camera.Parameters p = mCamera.getParameters();
setOptimalSize(p, w, h, SIZEOFPREVIEW);
setOptimalSize(p, w, h, SIZEOFPICTURE);
mCamera.setParameters(p);
mCamera.startPreview();
isPreviewRunning = true;
}
public void takePicture(){
Log.i(TAG+".takePicture", "being called!");
mCamera.takePicture(null, null, mPictureCallback);
Log.i(TAG+".takePicture", "call ended!");
}
}
CamLayer.takePicture()
将被外部类调用以启动。
问题是在Android 2.3.3中,takePicture
会挂起,因此会发现ANR问题。在/data/anr/traces.txt
中,找到以下内容。如您所见,native_takePicture
永远不会返回。
DALVIK THREADS:
(mutexes: tll=0 tsl=0 tscl=0 ghl=0 hwl=0 hwll=0)
"main" prio=5 tid=1 NATIVE
| group="main" sCount=1 dsCount=0 obj=0x40022170 self=0xce68
| sysTid=2411 nice=0 sched=0/0 cgrp=default handle=-1345006464
at android.hardware.Camera.native_takePicture(Native Method)
at android.hardware.Camera.takePicture(Camera.java:746)
at android.hardware.Camera.takePicture(Camera.java:710)
at oms.cj.tube.camera.CamLayer.takePicture(CamLayer.java:256)
at oms.cj.tube.camera.DefineColor.takePicture(DefineColor.java:61)
at oms.cj.tube.camera.DefineColor.onKeyUp(DefineColor.java:71)
at android.view.KeyEvent.dispatch(KeyEvent.java:1280)
at android.app.Activity.dispatchKeyEvent(Activity.java:2078)
at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:16
66)
at android.view.ViewRoot.deliverKeyEventToViewHierarchy(ViewRoot.java:2571)
at android.view.ViewRoot.handleFinishedEvent(ViewRoot.java:2546)
at android.view.ViewRoot.handleMessage(ViewRoot.java:1878)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:3691)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:507)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
at dalvik.system.NativeStart.main(Native Method)
有没有人有同样的问题?并知道如何解决它?
答案 0 :(得分:7)
我还观察到要冻结的mCamera.takePicture(null,null,handler)。我试着在调用takePicture()之前清除预览处理程序:mCamera.setPreviewCallback(null),现在它可以正常工作。
答案 1 :(得分:5)
今天我在使用Android 2.3.3在Samsung Exhibit 4G上测试我们的应用时遇到了完全相同的问题,并使用解决方法解决了它。
我不再使用 takepicture ,而是使用最后一次预览回调作为图片。
问题是预览回调使用NV21格式发送数据缓冲区。
所以你必须使用这个过程转换图像:NV21 - > RGB - >加载位图 - >压缩为JPEG
我们的代码现在看起来像这样:
camera.setPreviewCallback(new PreviewCallback() {
@Override
public synchronized void onPreviewFrame(byte[] data, Camera arg1) {
if (!mTakePicture) {
CameraPreview.this.invalidate();
} else {
if (mTakePictureCallback != null && !mPictureTaken) {
int rgb[] = new int[previewSize.width*previewSize.height];
decodeYUV420SP(rgb, data, previewSize.width, previewSize.height);
Bitmap memoryImage = Bitmap.createBitmap(rgb, previewSize.width, previewSize.height, Bitmap.Config.ARGB_8888);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
memoryImage.compress(CompressFormat.JPEG, 100, baos);
shutterSound();
setBackgroundDrawable(new BitmapDrawable(getContext().getResources(), memoryImage));
mTakePictureCallback.onPictureTaken(baos.toByteArray(), arg1);
}
mPictureTaken = true;
camera.stopPreview();
}
}
});
decodeYUV420SP 的代码在此处http://www.41post.com/3470/programming/android-retrieving-the-camera-preview-as-a-pixel-array在Ketai http://code.google.com/p/ketai/
找到了它当您拍照时,只需在 true
上设置 mTakePicture 变量我正在研究一个更好的版本,但这应该让你去。
答案 2 :(得分:1)
我不确定代码中使用的setOptimalSize方法是什么,但请确保已设置相机参数
mCamera.setPictureSize(captureSize.width, captureSize.height);
mCamera.setPictureFormat(ImageFormat.JPEG);
答案 3 :(得分:0)
问题在于下面的代码是写的。
定义了PreviewCallback,
PreviewCallback mPreviewCallback = new PreviewCallback() {
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
//Log.i(TAG+".mPreviewCallback.onPreviewFrame", "being called!");
}
};
mCamera.setPreviewCallback(mPreviewCallback);
适用于2.1 / 2.2,但不适用于2.3。
不确定Android团队是否支持这种使用相机的方式。 如果上述流程是预期的,那么Android团队应该解决这个问题。
答案 4 :(得分:0)
我用GB手机遇到了这个问题,对我来说,事实证明是因为我在调用camera.takePicture()之后立即调用了camera.startPreview(),这导致了Android中的一些线程锁定。修复是将camera.startPreview()移动到传递给camera.takePicture()的回调,这样只有在图片数据进入时才会调用它(下面的示例代码)。这当然只有在您拍摄照片后重新开始预览时才有意义。
// BAD BAD DON'T DO THIS!
public void myTakePicture(Camera.PictureCallback callback) {
mCamera.takePicture(null, null, null, callback);
mCamera.startPreview();
}
// ...
// The way that worked for me
public void myTakePicture(final Camera.PictureCallback callback) {
mCamera.takePicture(null, null, null, new Camera.PictureCallback() {
@Override
public void onPictureTaken(byte[] pictureData, Camera camera) {
callback.onPictureTaken(pictureData, camera);
mCamera.takePicture();
}
});
}
这不仅使得ANR在调用takePicture时消失了,而且还修复了在某些高端手机上发生的startPreview上的本机崩溃(特别是> = 4.3 Nexus 5)。希望它有所帮助!