安卓相机API应用在onResume时崩溃

5

我正在使用相机API编写一个应用程序:显示预览并拍照。第一个版本效果非常好。在Preview类中,继承自SurfaceView并实现了SurfaceHolder.Callback接口。

public void surfaceCreated(SurfaceHolder holder) {
    // The Surface has been created, acquire the camera and tell it where
    // to draw.
    camera = Camera.open();
    try {
        camera.setPreviewDisplay(holder);
        camera.startPreview();

    } catch (IOException e) {
        e.printStackTrace();
    }

    hasSurface = true;
}


public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
    // It will be called immediately after surfaceCreated
    // I move it to Resume.
    setCameraPreviewParameters();

    camera.startPreview();

}

在主活动中:
public class CameraDemo extends Activity

我还设置了一个成员变量: CameraUnit ui; 它是:

public class CameraUnit extends LinearLayout

在onCreate方法中,

    @Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    ui = new CameraUnit(this);
    setContentView((View) ui); 

    Log.d(TAG, "onCreate'd");
}

它创建了一个LinearLayout对象,其中包含SurfaceView和相机按钮。
onResume方法是:
    @Override
protected void onResume() {
    super.onResume();
}

虽然它可以工作,但我认为onResume不应该为空。我还看到surfaceCreated和surfaceChanged在著名的Android应用程序示例OCRTest中为空。因此,我认为将命令从surfaceCreated和surfaceChanged移动到onResume更好且可行。
我认为第一个版本的应用程序的流程图应该是: 主活动onCreate(创建LinearLayout及其surfaceView、相机按钮), 调用surfaceCreated和surfaceChanged, onResume。
因此,我可以简单地将surfaceCreated和surfaceChanged中的命令移动到onResume中。 但它不起作用! 我使用调试找出表面为空。我认为surfaceCreatd没有被调用。我的应用程序与OCRTest不同,在主活动中实现了SurfaceHolder.Callback。我在SurfaceView中实现了SurfaceHolder.Callback:
class Preview extends SurfaceView implements SurfaceHolder.Callback

所以我在onResume中添加了另外两行代码。
        ui.preview.surfaceCreated(ui.preview.mHolder);
    ui.preview.surfaceChanged(ui.preview.mHolder, 0, 800, 400);

我随机设置了最后三个整数参数。当我启动活动时,Android手机上弹出错误窗口。 它说“Caused by: java.lang.RuntimeException: Fail to connect to camera service”。 但是如何修复它?非常感谢!

错误信息如下:

    01-07 00:27:57.173: W/dalvikvm(11625): threadid=1: thread exiting with uncaught exception (group=0x4001d5a0)
01-07 00:27:57.173: E/AndroidRuntime(11625): FATAL EXCEPTION: main
01-07 00:27:57.173: E/AndroidRuntime(11625): java.lang.RuntimeException: Unable to resume activity {com.example/com.example.CameraDemo}: java.lang.RuntimeException: Fail to connect to camera service
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2460)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:2481)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1847)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.access$1500(ActivityThread.java:132)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1038)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.os.Handler.dispatchMessage(Handler.java:99)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.os.Looper.loop(Looper.java:150)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.main(ActivityThread.java:4263)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at java.lang.reflect.Method.invokeNative(Native Method)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at java.lang.reflect.Method.invoke(Method.java:507)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at dalvik.system.NativeStart.main(Native Method)
01-07 00:27:57.173: E/AndroidRuntime(11625): Caused by: java.lang.RuntimeException: Fail to connect to camera service
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.hardware.Camera.native_setup(Native Method)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.hardware.Camera.<init>(Camera.java:265)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.hardware.Camera.open(Camera.java:241)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at com.example.Preview.surfaceCreated(Preview.java:60)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at com.example.CameraDemo.onResume(CameraDemo.java:64)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1242)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.Activity.performResume(Activity.java:3904)
01-07 00:27:57.173: E/AndroidRuntime(11625):    at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2450)
01-07 00:27:57.173: E/AndroidRuntime(11625):    ... 12 more
01-07 00:27:59.455: I/Process(11625): Sending signal. PID: 11625 SIG: 9

我感觉这可能与Activity的onCreate、onResume的实现顺序以及surface的创建等有关。如果SurfaceView类实现了SurfaceHolder.Callback,即使在Activity的onResume点时,surface也还没有准备好。那么我们什么时候准备好呢?如果Activity实现了SurfaceHolder.Callback,那么内在的顺序是什么?救命啊!!! - user1914692
3个回答

5
你会得到这个。
java.lang.RuntimeException: Fail to connect to camera service

当相机被其他应用程序使用时,会出现异常。但我相信这个其他应用程序是您自己的应用程序。您需要释放相机。

以下是您应该释放相机的情况:

在预览类中:

public void surfaceDestroyed(SurfaceHolder holder) {

    // empty. Take care of releasing the Camera preview in your activity.
    if (mCamera != null) {
        mCamera.release();
    }
}

在活动中:

@Override
public void onBackPressed() {
    super.onBackPressed();
    if (myCamera != null) {
        myCamera.release();
    }
    finish();
}

@Override
protected void onPause() {
    // TODO Auto-generated method stub
    super.onPause();
    if (myCamera != null) {
        myCamera.release();
    }
}

当你点击“取消”时,以及当你完成使用相机时,即当图像被捕获并回到活动时,也会执行此操作。

还可以尝试以下方法:

public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {

    if (mHolder.getSurface() == null) {
        return;
    }

    try {
        mCamera.stopPreview();
    } catch (Exception e) {
        //You can ignore this, because this means the Preview doesn't Exist
        //So, no need to try stopping
    }

    try {
        mCamera.setPreviewDisplay(mHolder);
        mCamera.startPreview();
    } catch (Exception e) {
        //Catch this
    }
}

非常感谢。我已经在这些地方包含了相机释放功能。我的应用程序的第一个版本运行得非常好。正如我所说,我想将surfaceCreated和surfaceChanged中的命令移动到onResume中。问题是第一次调用onResume不起作用。因此,我猜测它可能与onResume和对surface的调用之间的关系有关。你的意见呢? - user1914692

1
我创建了一个应用程序,使用相机但不同的启动图标来进入设置页面。每次暂停并最小化应用程序以更改设置并返回时,它都会崩溃并显示此错误。经过大量研究,我仍然没有答案。然后我意识到这与相机的生命周期管理有关。因此,我查看了生命周期,特别是暂停。我意识到我们都在错误的方法上进行操作。我们应该在onStart上执行而不是在onResume上执行。然后它奏效了。
以下是我的应用程序代码。
@Override
    protected void onPause() {
        super.onPause();
        if (mCamera != null) {
            mCamera.stopPreview();
            mCamera.release();        // release the camera for other applications
            mCamera = null;
        }
        if (mPreview != null) {
            FrameLayout preview = (FrameLayout) findViewById(R.id.camera_viewer);
            preview.removeView(mPreview);
            mPreview = null;
        }
    }

    @Override
    protected void onStop() {
        super.onStop();
        if (mCamera != null) {
            mCamera.stopPreview();
            mCamera.release();        // release the camera for other applications
            mCamera = null;
        }
        if (mPreview != null) {
            FrameLayout preview = (FrameLayout) findViewById(R.id.camera_viewer);
            preview.removeView(mPreview);
            mPreview = null;
        }
    }
    @Override
    protected void onResume() {
        super.onResume();

    }

    @Override
    protected void onStart(){
        super.onStart();
//Check if the camera exists or not so it does not clash with the onCreate
        if(mCamera == null){
            dir_string =  new File("/storage/sdcard1/app");
            Log.d("TAG",dir_string.toString());
            mCamera = getCameraInstance();
            mPreview = new CameraPreview(this, mCamera);
            FrameLayout preview = (FrameLayout) findViewById(R.id.camera_viewer);
            preview.addView(mPreview);
        }
    }

希望这能至少帮助到某个人。

0

我发现在一些示例中,相机打开的代码既包含在surfaceChanged中,也包含在onResume中,使用if语句来避免重复。也许这是解决问题的一种方式。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接