Android相机源码.cameraSource.stop()导致应用程序冻结

14
我正在构建一个应用程序,使用Google Vision API具有QR扫描仪。在QR代码被读取后,我无法停止相机。流程是MainActivity-> QrActivity一旦QR码接收到检测,应用程序应返回到主活动。
如果我不调用cameraSource.release(),它可以正常工作,但设备会过热,并且对电池寿命有显着影响。但是,如果我释放相机源,则Mainactivity会变得无响应,应用程序将崩溃。
为什么变得无响应?释放相机源的正确位置在哪里? QrActivity
@Override
protected void onCreate(Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_qr);
    cancelBtn = (Button) findViewById(R.id.cancel_button);
    cancelBtn.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            onBackPressed();
        }
    });

    new QrReader(this);
}

QrReader类

public class QrReader {

    private static final String TAG = "QrReader";

    private SurfaceView cameraView;
    private TextView barcodeInfo;
    private BarcodeDetector barcodeDetector;
    private CameraSource cameraSource;
    private Activity mActivity;
    private AccessPointCredentials barCodeData;

    public QrReader(Activity activity) {
        this.mActivity = activity;

        cameraView = (SurfaceView) mActivity.findViewById(R.id.camera_view);
        barcodeInfo = (TextView) mActivity.findViewById(R.id.code_info);

        barcodeDetector =
                new BarcodeDetector.Builder(mActivity)
                        .setBarcodeFormats(Barcode.QR_CODE)
                        .build();

        cameraSource = new CameraSource
                .Builder(mActivity, barcodeDetector)
                .setAutoFocusEnabled(true)
                .build();

        cameraView.getHolder().addCallback(new SurfaceHolder.Callback() {

            @Override
            public void surfaceCreated(SurfaceHolder holder) {

                cameraSource = new CameraSource
                        .Builder(mActivity, barcodeDetector)
                        .setAutoFocusEnabled(true)
                        .setFacing(0)
                        .build();
                try {                  
                    cameraSource.start(cameraView.getHolder());

                } catch (Exception ioe) {
                    ioe.printStackTrace();
                }
            }

            @Override
            public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
            }

            @Override
            public void surfaceDestroyed(SurfaceHolder holder) {
                  // Log.i(TAG, "surfaceDestroyed: stopping camera Source");

                  // cameraSource.release();
            }
        });

        barcodeDetector.setProcessor(new Detector.Processor<Barcode>() {
            @Override
            public void release() {
                Log.i(TAG, "release: ");
            }

            @Override
            public void receiveDetections(Detector.Detections<Barcode> detections) {
                final SparseArray<Barcode> barCodes = detections.getDetectedItems();

                if (barCodes.size() != 0) {

                    Log.i(TAG, "received a Barcode");
                    barcodeInfo.post(new Runnable() {    // Use the post method of the TextView
                        public void run() {
                            barcodeInfo.setText(barCodes.valueAt(0).displayValue);

                        }
                    });
                    Gson g = new Gson();
                    try {
                        barCodeData = g.fromJson(barCodes.valueAt(0).rawValue, AccessPointCredentials.class);
                    } catch (Exception e) {
                        barCodeData = new AccessPointCredentials();
                        barCodeData.setSsid(barCodes.valueAt(0).rawValue);
                        barCodeData.setPass(null);
                        e.printStackTrace();
                    }

                    connectToWifi(barCodeData);

                    // CameraSource.release causes app to freeze

                    // cameraSource.release();
                }
            }
        });
    }

    private void connectToWifi(final AccessPointCredentials credentials) {

                //wificonnect code

    }

}

1
不确定,但请注意,如果您想进一步了解此问题,可以尝试使用CameraSource的开源副本:https://github.com/googlesamples/android-vision/blob/master/visionSamples/barcode-reader/app/src/main/java/com/google/android/gms/samples/vision/barcodereader/ui/camera/CameraSource.java#L322 - pm0733464
我尝试在两个被注释的位置(在surfaceDestroyed()receiveDetections()中)调用cameraSource.release() - Lonergan6275
2个回答

18

已经过去3个月了,但我刚刚遇到了同样的问题并解决了它。receiveDetections方法中的代码在不同的线程上运行,因此如果您想执行需要UI线程的操作,则需要从处理程序中发布它:

Handler handler = new Handler(Looper.getMainLooper());
handler.post(new Runnable() {
     @Override
     public void run() {
         cameraSource.release();
     }
});

谢谢你的回答,我有机会时会去查看。 - Lonergan6275
2
经过多个小时的令人沮丧的崩溃后,我确实可以确认这是一个线程问题。通过使用以下代码或使用view.post(new Runnable(){ cameraSource.release() }),您可以解决它。 - Spidfire

0

嘿哟,有更新吗?我遇到了一个类似的问题,我尝试处理那些release方法,但是不起作用。我只找到了一个快速修复/解决方法。希望有一个优雅的解决方案。

我的情况是,应用程序用户可以扫描QR码,然后进行付款。

MainActivity

  1. 抽屉菜单 (https://github.com/mikepenz/MaterialDrawer)
  2. 片段容器(当用户点击抽屉菜单时更改片段)

我所有其他的视图都是片段/ListFragment

ScanMainController

  1. 条形码检测器
  2. 相机源
  3. SurfaceView

我的情况是这样的,一旦我扫描了QR码,我就会推到另一个页面进行付款。但我需要找到一种方法来停止QR码扫描器,因为我只想让receiveDetections被调用一次。在iOS中,我曾经将delegate设置为nil,它与Android中的cameraSource.release();具有相同的功能。但是,我无法找到一个适当的位置来调用release()方法。

如果我主动调用release(),那么当用户通过点击抽屉菜单导航到其他页面时,我的FragmentTransaction提交将被整个应用程序冻结。然后,Android系统会弹出告诉我应用程序未响应。

我通过在“receiveDetections”中添加一个“boolean”标志来快速修复,以确保我只推送视图一次。它有效。但我正在寻找一种更优雅和“正确”的方法来做到这一点。

谢谢


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