Android相机中的媒体录制器启动失败

8

我正在开发一个视频录制应用程序,在其中我想显示预览,当用户点击录制按钮时,它开始录制,当用户点击停止按钮时,它停止录制。

我在我的界面上得到了视频预览,但是当我按下开始按钮时,它会崩溃并显示错误“MEDIA.RECORDER.START(Native MEthod)。”这里是我的代码,请帮助我。

  @Override
    public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            surfaceView = (SurfaceView) findViewById(R.id.surface_camera);
            surfaceHolder = surfaceView.getHolder();
            surfaceHolder.addCallback(this);
            surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
            Button start =(Button)findViewById(R.id.start);
            Button stop =(Button)findViewById(R.id.stop);
            stop.setOnClickListener(new OnClickListener() {

                @Override
                public void onClick(View v) {
                    // TODO Auto-generated method stub
                    camera.stopPreview();
                    stopRecording();
                }
            });
            start.setOnClickListener(new OnClickListener() {

                @Override
                public void onClick(View v) {
                    // TODO Auto-generated method stub
                    startRecording();
                }
            });
    }

    @Override
public void surfaceCreated(SurfaceHolder holder) {
    camera = Camera.open();
    if (camera != null){
        Camera.Parameters params = camera.getParameters();
        camera.setParameters(params);
    }
    else {
        Toast.makeText(getApplicationContext(), "Camera not available!", Toast.LENGTH_LONG).show();
        finish();
    }
}

@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    if (previewRunning){
        camera.stopPreview();
    }
    Camera.Parameters p = camera.getParameters();
     List<Camera.Size> sizes = p.getSupportedPreviewSizes(); 
     Camera.Size cs = sizes.get(0);  
     p.setPreviewSize(cs.width, cs.height);  
       camera.setParameters(p);

    try {
        camera.setPreviewDisplay(holder);
        camera.startPreview();
        previewRunning = true;
    }
    catch (IOException e) {
        Log.e(TAG,e.getMessage());
        e.printStackTrace();
    }
}

 private MediaRecorder mediaRecorder;
    private final int maxDurationInMs = 20000;
    private final long maxFileSizeInBytes = 500000;
    private final int videoFramesPerSecond = 20;

    public boolean startRecording(){
        try {
            camera.unlock();

            mediaRecorder = new MediaRecorder();

            mediaRecorder.setCamera(camera);
            mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
            mediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);

            mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT);

            mediaRecorder.setMaxDuration(maxDurationInMs);

        File tempFile = new File(getCacheDir(),"test.mp4");
            mediaRecorder.setOutputFile(tempFile.getPath());

            mediaRecorder.setVideoFrameRate(videoFramesPerSecond);
            mediaRecorder.setVideoSize(surfaceView.getWidth(), surfaceView.getHeight());

            mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
            mediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.DEFAULT);

            mediaRecorder.setPreviewDisplay(surfaceHolder.getSurface());

            mediaRecorder.setMaxFileSize(maxFileSizeInBytes);

                        mediaRecorder.prepare();
            mediaRecorder.start();

            return true;
        } catch (IllegalStateException e) {
            Log.e(TAG,e.getMessage());
            e.printStackTrace();
            return false;
        } catch (IOException e) {
            Log.e(TAG,e.getMessage());
            e.printStackTrace();
            return false;
        }
    }

    public void stopRecording(){
        mediaRecorder.stop();
        camera.lock();
    }

@Override
public void surfaceDestroyed(SurfaceHolder holder) {
    camera.stopPreview();
    previewRunning = false;
    camera.release();
}

}

日志是

 08-31 02:20:11.781: E/MediaRecorder(14519): start failed: -19
 08-31 02:20:11.781: D/AndroidRuntime(14519): Shutting down VM
 08-31 02:20:11.781: W/dalvikvm(14519): threadid=1: thread exiting with uncaught exception (group=0x416c9700)
 08-31 02:20:11.781: E/AndroidRuntime(14519): FATAL EXCEPTION: main
 08-31 02:20:11.781: E/AndroidRuntime(14519): java.lang.RuntimeException: start failed.
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.media.MediaRecorder.start(Native Method)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at com.example.cameratest.MainActivity.startRecording(MainActivity.java:135)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at com.example.cameratest.MainActivity$2.onClick(MainActivity.java:61)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.view.View.performClick(View.java:4240)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.view.View$PerformClick.run(View.java:17721)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.os.Handler.handleCallback(Handler.java:730)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.os.Handler.dispatchMessage(Handler.java:92)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.os.Looper.loop(Looper.java:137)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at android.app.ActivityThread.main(ActivityThread.java:5103)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at java.lang.reflect.Method.invokeNative(Native Method)
08-31 02:20:11.781: E/AndroidRuntime(14519):    at java.lang.reflect.Method.invoke(Method.java:525)
08-31 02:20:11.781: E/AndroidRuntime(14519):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
08-31 02:20:11.781: E/AndroidRuntime(14519):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
 08-31 02:20:11.781: E/AndroidRuntime(14519):   at dalvik.system.NativeStart.main(Native Method)

完整的 Log Cat 管理器... - TheLittleNaruto
@KumarGaurav 这是完整的日志记录.. - user1597878
你尝试过删除 mediaRecorder.setVideoFrameRate(videoFramesPerSecond); 这行代码吗? - TheLittleNaruto
@KumarGaurav,它不起作用,是的,我尝试过了。 - user1597878
6个回答

12

我发现了这个解决方案,对我有用,并且解决了这个问题 :)

对于2.3+的安卓版本:

只需更改:

mediaRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));

致:

mediaRecorder.setProfile(CamcorderProfile.get(CameraID,CamcorderProfile.QUALITY_HIGH));

不错!如果您没有使用默认摄像头,则应将此内容添加到开发人员页面,以指定摄像头 ID! - Billda

3
在您的startRecording方法中,尝试在解锁之前锁定相机:
mediaRecorder = new MediaRecorder();
camera.lock();
camera.unlock();

2

注释此代码行,

mediaRecorder.setVideoSize(surfaceView.getWidth(), surfaceView.getHeight());

为什么?我在设置VideoSize时遇到了问题。http://stackoverflow.com/questions/33898173/android-media-recorder-setvideosize-do-not-work - Anton Shkurenko
谢谢兄弟!实际上,这应该是支持的相机预览尺寸之一。 - SaadurRehman

0

尝试以下方法,也许会有帮助:

        try {
            mediaRecorder.prepare();
            Thread.sleep(1000);
        } catch (Exception e) {
            // TODO: handle exception
            e.printStackTrace();
        }
        mediaRecorder.start();

对于来电,准备需要花费一些时间。

等待1秒钟进行准备,一切都可以正常工作。


0

-3

这可能比你想象的要简单得多。Android已经为像录制和返回视频文件这样简单的事情做了很多工作,使用Intents...

使用相机应用程序录制视频 将操作委托给其他应用程序的Android方式是调用描述所需操作的Intent。这涉及三个部分:Intent本身,调用外部Activity的调用以及一些代码来处理当焦点返回到您的活动时的视频。

下面是一个调用意图以捕获视频的函数。

private void dispatchTakeVideoIntent() {
Intent takeVideoIntent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
startActivityForResult(takeVideoIntent, ACTION_TAKE_VIDEO);
}

观看视频 Android相机应用程序将视频作为指向存储位置的Uri返回到onActivityResult()中。以下代码检索此视频并在VideoView中显示它。

private void handleCameraVideo(Intent intent) {
mVideoUri = intent.getData();
mVideoView.setVideoURI(mVideoUri);
}

将此添加到清单中

<uses-feature android:name="android.hardware.camera" />

来源:http://developer.android.com/training/camera/videobasics.html


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