14

我正在使用Android4OpenCV进行一些实时图像处理,并且我想使用相机可以提供的最小分辨率。默认分辨率是相机可以提供的最大分辨率。

我正在查看第三个示例,它允许用户通过菜单更改分辨率。我想修改该示例以在启动时更改分辨率,而不是要求用户通过菜单。为此,我只需在原本为空的onCameraViewStarted()函数中添加两行:

public void onCameraViewStarted(int width, int height) {
    android.hardware.Camera.Size res = mOpenCvCameraView.getResolutionList().get(mOpenCvCameraView.getResolutionList().size()-1);
    mOpenCvCameraView.setResolution(res);
}

问题是,这在我运行 Android 4.2.2 的 Galaxy Nexus 上运行良好。应用程序启动,并且分辨率设置正确。

但是,当我在运行 Android 5.1 的 Nexus 7 平板电脑上运行完全相同的应用程序时,该应用程序在调用setResolution(). 实际上它一次可以正常工作,但第二次尝试运行它时会挂起 - 即使您完全退出应用程序,将其从正在运行的应用程序中删除,或重新启动设备。其他用户也报告了同样的错误,所以不仅仅是 Nexus 7 设备——事实上,我的 Galaxy Nexus 似乎是唯一可以使用的设备。

具体来说,应用程序进入setResolution()函数,然后调用org.opencv.android.JavaCameraView.disconnectCamera(),如下所示:

(注意:此代码是 OpenCV4Android 库的内部代码,这不是我的代码)

protected void disconnectCamera() {
    /* 1. We need to stop thread which updating the frames
     * 2. Stop camera and release it
     */
    Log.d(TAG, "Disconnecting from camera");
    try {
        mStopThread = true;
        Log.d(TAG, "Notify thread");
        synchronized (this) {
            this.notify();
        }
        Log.d(TAG, "Wating for thread");
        if (mThread != null)
            mThread.join();
    } catch (InterruptedException e) {
        e.printStackTrace();
    } finally {
        mThread =  null;
    }

    /* Now release camera */
    releaseCamera();
}

查看日志,我可以看到线程卡在了synchronized(this)线上。唯一在该 Object 上同步的是内部JavaCameraView.CameraWorker类,它是mThread上面代码中的变量,由 JavaCameraView 类启动:

(注意:此代码是 OpenCV4Android 库的内部代码,这不是我的代码)

private class CameraWorker implements Runnable {

    public void run() {
        do {
            synchronized (JavaCameraView.this) {
                try {
                    JavaCameraView.this.wait();
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }

            if (!mStopThread) {
                if (!mFrameChain[mChainIdx].empty())
                    deliverAndDrawFrame(mCameraFrame[mChainIdx]);
                mChainIdx = 1 - mChainIdx;
            }
        } while (!mStopThread);
        Log.d(TAG, "Finish processing thread");
    }
}

我已经尝试过使用该代码,将 notify() 更改为 notifyAll(),并维护一个 CameraWorker 线程列表并加入每个线程。但无论如何,该应用程序仍然挂起disconnectCamera()电话。

我的问题是:

  • 如何修改第三个 OpenCV4Android 示例,以便在启动时设置其分辨率?

  • 是什么导致应用程序挂起?

  • 为什么这在某些设备上有效,而在其他设备上无效?

编辑:我没有收到任何评论或答案,所以我在这里交叉发布到 OpenCV 论坛。

编辑2:根据cyriel的建议,我尝试在几帧过去后设置分辨率:

int frames = 0;
public Mat onCameraFrame(CvCameraViewFrame inputFrame) {

    frames++;

    if(frames == 6){
        android.hardware.Camera.Size res = mOpenCvCameraView.getResolutionList().get(mOpenCvCameraView.getResolutionList().size()-1);
        mOpenCvCameraView.setResolution(res);

    }

    return inputFrame.rgba();
}

但是,现在它卡在了同一个地方,即使在我的 Galaxy Nexus 上,如果我在onCameraViewStarted()函数中设置分辨率,它也可以工作。我尝试将帧数增加到 7 甚至 100,但我总是卡在同一个地方。

4

1 回答 1

5

在您的情况下,最重要的问题是,如果您根本不修改代码,它是否可以工作 - 您是否能够更改分辨率(通过菜单)而不会使应用程序崩溃?
如果是,那么答案很可能很简单——OpenCV 中的错误与 Windows 版本中的错误相同:在更改相机分辨率或 fps(很可能是任何属性)之前,您需要至少获取一个(确定使用 3-5)更改此属性之前的框架。
如果没有,您很可能无能为力 - 填写错误报告并等待评论。唯一的选择是使用其他库从相机中抓取帧,然后将其转换为 OpenCV 对象。

于 2015-05-11T22:31:10.617 回答