Android cameraSource.stop() 导致应用冻结
Android cameraSource.stop() causing app to freeze
我正在使用 google 视觉 api 构建一个具有二维码扫描仪的应用程序。读取二维码后我无法停止相机。流量是 MainActivity -> QrActivity
一旦二维码收到检测,应用程序应该 return 到主 activity.
如果我不调用 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 Class
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
}
}
已经 3 个月了,但我偶然发现了同样的问题并解决了这个问题。 receiveDetections 方法中的代码是 运行 在不同的线程上,所以如果你想做一些需要 ui 线程的事情,你需要从处理程序 post 它:
Handler handler = new Handler(Looper.getMainLooper());
handler.post(new Runnable() {
@Override
public void run() {
cameraSource.release();
}
});
嘿哟,有更新吗?我刚遇到类似的问题,我尝试处理那些 release
方法,但不起作用。我只找到了一个快速修复/解决方法。想有一个优雅的解决方案。
我的情况是,应用程序用户可以扫描二维码然后付款。
MainActivity
- 抽屉菜单(https://github.com/mikepenz/MaterialDrawer)
- 片段容器(当用户点击抽屉菜单时改变片段)
我的其他观点都是Fragment/ListFragment
ScanMainController
- 条码检测器
- 相机源
- 表面视图
我的场景是这样的,扫描二维码后,我推送到另一个页面用于支付。但是我需要找到一种方法来停止 QRCode 扫描仪,因为我只希望 receiveDetections
只被调用一次。在iOS中,我曾经设置delegate to nil
,它与Android中的cameraSource.release();
做同样的工作。但是,我就是找不到合适的位置来调用 release()
方法。
如果我主动调用 release()
,那么当用户想要通过单击抽屉菜单导航到其他页面时,我的 FragmentTransaction 提交将被整个应用程序冻结触发。然后 Android 系统会弹出告诉我应用程序没有响应。
我通过在 'receiveDetections' 中添加一个 'boolean' 标志来快速修复,以保证我只推送视图一次。有用。但我正在寻找一种更优雅的方式和 'proper' 方式来做到这一点。
谢谢
我正在使用 google 视觉 api 构建一个具有二维码扫描仪的应用程序。读取二维码后我无法停止相机。流量是 MainActivity -> QrActivity
一旦二维码收到检测,应用程序应该 return 到主 activity.
如果我不调用 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 Class
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
}
}
已经 3 个月了,但我偶然发现了同样的问题并解决了这个问题。 receiveDetections 方法中的代码是 运行 在不同的线程上,所以如果你想做一些需要 ui 线程的事情,你需要从处理程序 post 它:
Handler handler = new Handler(Looper.getMainLooper());
handler.post(new Runnable() {
@Override
public void run() {
cameraSource.release();
}
});
嘿哟,有更新吗?我刚遇到类似的问题,我尝试处理那些 release
方法,但不起作用。我只找到了一个快速修复/解决方法。想有一个优雅的解决方案。
我的情况是,应用程序用户可以扫描二维码然后付款。
MainActivity
- 抽屉菜单(https://github.com/mikepenz/MaterialDrawer)
- 片段容器(当用户点击抽屉菜单时改变片段)
我的其他观点都是Fragment/ListFragment
ScanMainController
- 条码检测器
- 相机源
- 表面视图
我的场景是这样的,扫描二维码后,我推送到另一个页面用于支付。但是我需要找到一种方法来停止 QRCode 扫描仪,因为我只希望 receiveDetections
只被调用一次。在iOS中,我曾经设置delegate to nil
,它与Android中的cameraSource.release();
做同样的工作。但是,我就是找不到合适的位置来调用 release()
方法。
如果我主动调用 release()
,那么当用户想要通过单击抽屉菜单导航到其他页面时,我的 FragmentTransaction 提交将被整个应用程序冻结触发。然后 Android 系统会弹出告诉我应用程序没有响应。
我通过在 'receiveDetections' 中添加一个 'boolean' 标志来快速修复,以保证我只推送视图一次。有用。但我正在寻找一种更优雅的方式和 'proper' 方式来做到这一点。
谢谢