android 设置录制视频的时长与大小限制,超时、超大小时自动停止

本文介绍了一个基于Android的应用示例,通过使用MediaRecorder和定时器实现对录制视频的时长及文件大小进行限制。文章详细展示了如何利用SharedPreferences保存设置,并通过SeekBars让用户调整最大录制时间和文件大小。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

项目中要用到录制视频,需要对视频的大小、录制时长做限制。所以开发了此demo。

1、原理很简单,对于超时控制:做个定时器,到时间自动停止。

2、对于文件大小控制:还是使用定时器,每个一定时间检测文件大小,超过设置大小自动停止。

本demo时长、文件大小设置使用的是SharedPreferences

 

package com.lbl;


import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.DialogInterface.OnShowListener;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.hardware.Camera;
import android.hardware.Camera.Parameters;
import android.hardware.Camera.Size;
import android.media.MediaRecorder;
import android.media.MediaRecorder.OnInfoListener;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.view.SurfaceView;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.SeekBar;
import android.widget.SeekBar.OnSeekBarChangeListener;
import android.widget.TextView;
import android.widget.Toast;

import com.example.myandroiddemo.R;

public class RecordVideo extends Activity{
	// 程序中的两个按钮
	private Button record , stop;
	TextView currentTimeLengthTextView, currentFileSizeTextView;
	
	// 系统的视频文件
	private File videoFile ;
	private MediaRecorder mRecorder;
	// 显示视频预览的SurfaceView
    SurfaceView sView;
    // 记录是否正在进行录制
    private boolean isRecording = false;
    private Handler handler;
    
    //控制时长seekBar
    private SeekBar timeLengthSeekBar;
    //控制文件大小seekBar
    private SeekBar fileSizeSeekBar;
    
    SharedPreferences sharedPreferences;
    private Camera camera;
    
    
    //控制时长定时器
    private Timer timer_timeLength;
    //控制文件大小定时器
    private Timer timer_fileSize;

    @Override
	public void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.media_recorder);
		record = (Button) findViewById(R.id.record);
		stop = (Button) findViewById(R.id.stop);
		
		// 让stop按钮不可用。
		stop.setEnabled(false);
		// 为两个按钮的单击事件绑定监听器
		record.setOnClickListener(new OnButtonClick());
		stop.setOnClickListener(new OnButtonClick());
		
		// 获取程序界面中的SurfaceView
		sView = (SurfaceView) this.findViewById(R.id.sView);
		/**
		 * 出现这些问题基本上都是以下两个方法参数导致的,每个手机的分辨率和预览大小支持都不一样,设置错误就会报错。
		 */
//        sView.getHolder().setFixedSize(320, 280);
        // 设置该组件让屏幕不会自动关闭
        sView.getHolder().setKeepScreenOn(true);
        
        sharedPreferences = getSharedPreferences("settings", MODE_PRIVATE);
        
        handler = new Handler(){
        	@Override
        	public void handleMessage(Message msg) {
        		if(msg.what == 0x123){
        			stop();
        			Toast.makeText(RecordVideo.this, "录制视频由于已超过最长时间,自动停止!", 200).show();
        		}else if(msg.what == 0x124){
        			stop();
        			Toast.makeText(RecordVideo.this, "录制视频由于已超过设置的最大文件大小,自动停止!", 200).show();
        		}
        	}
        };
	}
    
    class OnButtonClick implements OnClickListener{

		@Override
		public void onClick(View v) {
			switch (v.getId())
			{
				// 单击录制按钮
				case R.id.record:
					if (!Environment.getExternalStorageState().equals(
						android.os.Environment.MEDIA_MOUNTED)){
						Toast.makeText(RecordVideo.this
							, "SD卡不存在,请插入SD卡!"
							, Toast.LENGTH_SHORT).show();
						return;
					}
					
					init();
					start();				
					//启动停止录制的定时器
					timer_timeLength = new Timer();
					System.out.println();
					System.out.println(sharedPreferences.getInt("maxTimeLength", 0) * 1000);
					timer_timeLength.schedule(new EndTimerThread(), sharedPreferences.getInt("maxTimeLength", 0) * 1000);
					
					
					//每个1秒输出视频大小
//					timer_fileSize = new Timer();
//					timer_fileSize.schedule(new CheckFileSizeThread(), new Date(), 500);
					
					break;
				// 单击停止按钮
				case R.id.stop:
					// 如果正在进行录制
					stop();
					break;
			}
		}
    	
    }

    class OnSeekBarChange implements OnSeekBarChangeListener{

		@Override
		public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
			if(seekBar.getId() == R.id.timeLengthSeekBar){
				currentTimeLengthTextView.setText("" + progress);
			}else{
				currentFileSizeTextView.setText("" + progress);
			}
		}

		@Override
		public void onStartTrackingTouch(SeekBar seekBar) {
		}

		@Override
		public void onStopTrackingTouch(SeekBar seekBar) {
		}
    }

	
	/**
	 * 初始化MediaRecoder
	 * 参数设置顺序很重要,否则出错
	 * @throws IOException 
	 */
	private void init(){
		// 创建保存录制视频的视频文件
		try {
			videoFile = new File(Environment.getExternalStorageDirectory().getCanonicalFile() + "/myvideo.mp4");
		} catch (IOException e) {
			e.printStackTrace();
		}
		// 创建MediaPlayer对象
		mRecorder = new MediaRecorder();
		mRecorder.reset();
		
		camera = Camera.open();
		camera.setDisplayOrientation(90);
		
        Parameters parameters = camera.getParameters();
//        List<Size> supportedPictureSizes = parameters.getSupportedPictureSizes();
//        System.out.println("supportedPictureSizes================");
//        for (Size size : supportedPictureSizes) {
//			System.out.println( "width: " + size.width + ";height:" + size.height);
//		}
//        List<Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes();
//        System.out.println("supportedPreviewSizes================");
//        for (Size size : supportedPreviewSizes) {
//			System.out.println( "width: " + size.width + ";height:" + size.height);
//		}
//        List<Size> supportedVideoSizes = parameters.getSupportedVideoSizes();
//        System.out.println("supportedVideoSizes================");
//        for (Size size : supportedVideoSizes) {
//			System.out.println( "width: " + size.width + ";height:" + size.height);
//		}
        
        
        camera.unlock();
        mRecorder.setCamera(camera);
        
		// 设置从麦克风采集声音
		mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
		// 设置从摄像头采集图像
		mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
		// 设置视频文件的输出格式
		// 必须在设置声音编码格式、图像编码格式之前设置
		mRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
		//构造CamcorderProfile,使用高质量视频录制
//		CamcorderProfile camcorderProfile = CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH);
//		mRecorder.setProfile(camcorderProfile);
		// 设置声音编码的格式
		mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
		// 设置图像编码的格式
		mRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.MPEG_4_SP);
		/**
		 *设置视频录制的分辨率。必须放在设置编码和格式的后面,否则报错
		 *且这里的分辨率必须是该camera支持的
		 */
        List<Size> supportedVideoSizes = parameters.getSupportedVideoSizes();
        long maxFileSize = Long.parseLong(200 * 1024 + "");//如果获取不到视频分辨率信息则默认设置个100k
        
        if(null != supportedVideoSizes && supportedVideoSizes.size() > 0){
        	mRecorder.setVideoSize(supportedVideoSizes.get(0).width, supportedVideoSizes.get(0).height);
        	maxFileSize = Long.parseLong(sharedPreferences.getInt("maxFileSize", 0) * 1024 * 1024 + "");
        }else{
        	Toast.makeText(RecordVideo.this, "无法获取视频分辨率信息,此处默认最大200K。", 200).show();
        }
        
		// 每秒 4帧。。设置录制的视频帧率。必须放在设置编码和格式的后面,否则报错
		mRecorder.setVideoFrameRate(15);
		
		System.out.println("maxFileSize:"+maxFileSize);
		mRecorder.setMaxFileSize(maxFileSize);
		mRecorder.setOnInfoListener(new OnInfoListener() {
			
			@Override
			public void onInfo(MediaRecorder mr, int what, int extra) {
				if(what == MediaRecorder.MEDIA_RECORDER_INFO_MAX_FILESIZE_REACHED){
					stop();
        			Toast.makeText(RecordVideo.this, "录制视频由于已超过设置的最大文件大小,自动停止!", 200).show();
				}
				
			}
		});
		
		
		mRecorder.setOutputFile(videoFile.getAbsolutePath());
		
		
		
		// 指定使用SurfaceView来预览视频
		mRecorder.setPreviewDisplay(sView.getHolder().getSurface());
	}
	
	
	/**
	 * 开始录制视频
	 * @throws IOException 
	 * @throws IllegalStateException 
	 */
	private void start(){
		if(!isRecording){
			try {
				mRecorder.prepare();
			} catch (IllegalStateException e) {
				e.printStackTrace();
			} catch (IOException e) {
				e.printStackTrace();
			}
			// 开始录制
			mRecorder.start();
			// 让record按钮不可用。
			record.setEnabled(false);
			// 让stop按钮可用。
			stop.setEnabled(true);
			isRecording = true;
		}else{
			Toast.makeText(RecordVideo.this, "您正在录制,请先停止录制视频!", 200).show();
		}
	}
	
	/**
	 * 停止录制视频
	 */
	private void stop(){
		if (isRecording){
			// 停止录制
			mRecorder.stop();
			// 释放资源
			mRecorder.release();
			mRecorder = null;
			
			camera.stopPreview();
			camera.release();
			camera = null;
			
			// 让record按钮可用。
			record.setEnabled(true);
			// 让stop按钮不可用。
			stop.setEnabled(false);
			isRecording = false;
		}else{
//			Toast.makeText(RecordVideo.this, "对不起,您还没有开始录制视频!", 200).show();
		}
		
		if(null != timer_timeLength){
			timer_timeLength.cancel();
		}
		if(null != timer_fileSize){
			timer_fileSize.cancel();
		}
	}
	
	
	
	/**
	 * 停止录像定时器
	 */ 
	class EndTimerThread extends TimerTask {
		@Override  
		public void run() {  
			handler.sendEmptyMessage(0x123);
			this.cancel(); 
		} 
	}
	
	
//	/**
//	 * 检测文件大小,超过设置的最大值时停止录制
//	 */ 
//	class CheckFileSizeThread extends TimerTask {
//		@Override  
//		public void run() {  
//			System.out.println("videoFile.length(): "+videoFile.length() + "; sharedPreferences.maxFileSize: " + sharedPreferences.getInt("maxFileSize", 0));
//			System.out.println(videoFile.length()/1024/1024);
//			
//			if(videoFile.length()/1024/1024 >= sharedPreferences.getInt("maxFileSize", 0)){
//				handler.sendEmptyMessage(0x124);
//				this.cancel();
//			}
//		} 
//	}
	
	/**
	 * 设置按钮单机事件
	 * @param v
	 */
	public void setting(View v){
		LinearLayout linearLayout = (LinearLayout)getLayoutInflater().inflate(R.layout.setting, null);
		
		AlertDialog alertDialog = new AlertDialog.Builder(RecordVideo.this).setTitle("设置").setIcon(R.drawable.ic_launcher)
			.setView(linearLayout)
			.setPositiveButton("确定", new DialogInterface.OnClickListener() {
	
				@Override
				public void onClick(DialogInterface dialog, int which) {
					//保存数据到SharedPreferences中
					Editor editor = sharedPreferences.edit();
					editor.putInt("maxTimeLength", timeLengthSeekBar.getProgress());
					editor.putInt("maxFileSize", fileSizeSeekBar.getProgress());
					
					//提交数据
					editor.commit();
				}
			})
			.create();
		
		//弹出设置对话框时初始化保存的试着信息
		alertDialog.setOnShowListener(new OnShowListener() {
			
			@Override
			public void onShow(DialogInterface dialog) {
				currentTimeLengthTextView.setText(sharedPreferences.getInt("maxTimeLength", 0) + "");
				currentFileSizeTextView.setText(sharedPreferences.getInt("maxFileSize", 0) + "");
				
				timeLengthSeekBar.setProgress(sharedPreferences.getInt("maxTimeLength", 0));
				fileSizeSeekBar.setProgress(sharedPreferences.getInt("maxFileSize", 0));
			}
		});
		
		alertDialog.show();
		
		//获取设置界面的元素
		timeLengthSeekBar = (SeekBar)linearLayout.findViewById(R.id.timeLengthSeekBar);
		fileSizeSeekBar = (SeekBar)linearLayout.findViewById(R.id.fileSizeSeekBar);
		currentTimeLengthTextView = (TextView) linearLayout.findViewById(R.id.currentTimeLength);
		currentFileSizeTextView = (TextView) linearLayout.findViewById(R.id.currentFileSize);
		
		
		//给seekBar添加监听器
		timeLengthSeekBar.setOnSeekBarChangeListener(new OnSeekBarChange());
		fileSizeSeekBar.setOnSeekBarChangeListener(new OnSeekBarChange());
	}
}

 

media_recorder.xml文件:

 

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
	xmlns:android="http://schemas.android.com/apk/res/android"
	android:orientation="vertical"
	android:layout_width="match_parent"
	android:layout_height="match_parent">
<!-- 显示视频预览的SurfaceView -->
<SurfaceView
	android:id="@+id/sView"
	android:layout_width="match_parent"
	android:layout_height="match_parent"
	 />
<LinearLayout
	android:orientation="horizontal"
	android:layout_width="wrap_content"
	android:layout_height="wrap_content"
	android:gravity="center_horizontal"
	android:layout_alignParentBottom="true"
	android:layout_centerHorizontal="true">
	<Button 
	    android:id="@+id/settintg"
	    android:layout_width="wrap_content"
	    android:layout_height="wrap_content"
	    android:text="设置"
	    android:onClick="setting"
	    />
	<Button
		android:id="@+id/record"
		android:layout_width="wrap_content"
		android:layout_height="wrap_content"
		android:text="开始"/>
	<Button
		android:id="@+id/stop"
		android:layout_width="wrap_content"
		android:layout_height="wrap_content"
		android:text="结束"/>
</LinearLayout>
</RelativeLayout>


 

AndroidManifest.xml文件中别忘了加权限哦:

<uses-permission android:name="android.permission.RECORD_AUDIO"/>
    <uses-permission android:name="android.permission.CAMERA"/>
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>


 

<think>嗯,我现在遇到了一个Android Camera2 CTS测试失败的问题,具体是RecordingTest里的testConstrainedHighSpeedRecording方法失败了,错误信息是“Wait for a capture start timed out in 3000ms”。我需要弄清楚为什么会发生这个超时问题,以及如何解决它。 首先,我得了解这个测试的目的。Constrained High Speed Recording应该是测试设备在高速录像模式下的表现,可能涉及到高帧率的视频录制,比如慢动作视频。根据Android的文档,Camera2的High Speed Capture API允许应用以高帧率捕获视频,比如120fps或更高。测试失败可能意味着设备在启动高帧率录制时出了问题,导致在3秒内没有收到capture开始的信号。 错误堆栈显示在CameraTestUtils.java的1268行调用了assertNotNull,而SimpleCaptureCallback的getCaptureStartTimestamps方法可能返回了null,说明没有收到预期的CaptureStart事件。这可能是因为相机子系统没有正确启动捕获,或者在规定时间内没有响应。 接下来,我需要考虑可能的原因。首先想到的是硬件兼容性问题,比如设备不支持特定的高帧率配置。或者,相机驱动或固件有问题,导致无法及时响应请求。软件层面,可能是测试用例中的配置参数不正确,比如帧率设置超出了设备能力,或者surface配置不当。 另外,测试环境也可能有影响。比如设备在测试时处于繁忙状态,CPU或内存资源不足,导致相机服务无法及时处理请求。还有可能是其他应用占用了相机资源,导致测试无法正常获取相机设备。 我应该查阅相关的Android CTS文档,看看testConstrainedHighSpeedRecording的具体要求和条件。可能需要检查设备的相机能力,确认是否支持所需的高帧率模式。例如,通过CameraCharacteristics获取支持的High Speed Video配置,确认测试使用的配置是否在支持列表中。 另外,检查测试代码中的参数设置是否正确。比如使用的视频尺寸、帧率、以及是否正确地创建了录制会话(Session)和配置了输出Surface。如果配置错误,相机可能无法启动录制,导致超时。 还有可能是在创建会话或提交请求时出现了问题,比如重复使用了已关闭的会话,或者请求队列被填满而没有及时处理。需要确保测试代码正确管理了相机的会话和请求,特别是在高帧率场景下的资源管理。 另一个方面是权限问题,测试可能需要特定的权限来访问高帧率模式,或者设备上的某些设置限制了相机的功能。例如,省电模式可能会影响相机的性能,导致无法达到所需的帧率。 此外,查看日志可能会有帮助,比如logcat输出中的相机相关错误或警告,可能能提供更详细的失败原因。例如,是否有相机服务报错,或者底层驱动返回了错误码。 如果问题是偶发的,可能是由于超时时间设置过短,在部分设备上启动高帧率录制需要更长的时间。可以尝试调整测试中的等待时间,看是否能够解决问题。但作为CTS测试,可能无法直接修改超时参数,这时候需要确认设备是否符合CTS的要求。 最后,可能需要联系设备厂商或查看已知问题列表,确认是否存在已知的兼容性问题或固件缺陷,导致该测试失败。如果是硬件限制,可能需要豁免该测试,但CTS通常要求必须通过所有测试项才能认证。</think>针对 `android.hardware.camera2.cts.RecordingTest#testConstrainedHighSpeedRecording` 的测试失败问题,以下是逐步分析解决方案: --- ### **1. 问题背景** - **测试目的**:验证设备是否支持**Constrained High-Speed Recording**(受限高速录制),即高帧率视频录制(如慢动作视频)。 - **失败现象**:超时错误 `Wait for a capture start timed out in 3000ms`,表示相机未在3秒内启动录制。 --- ### **2. 错误根源分析** #### **直接原因** - **未收到 `CaptureStart` 事件**:测试代码通过 `SimpleCaptureCallback.getCaptureStartTimestamps()` 等待录制开始的信号,但未在超时时间内收到。 #### **潜在原因** 1. **硬件限制** - 设备未支持测试要求的高帧率配置(如120fps、240fps)。 2. **配置错误** - 测试使用的分辨率、帧率或格式超出设备能力。 - 未正确配置相机会话(`CameraCaptureSession`)或输出Surface。 3. **资源冲突** - 其他进程占用相机资源,或设备CPU/内存负载过高。 4. **固件/驱动问题** - 相机驱动未正确处理高帧率请求。 5. **超时时间不足** - 某些设备启动高帧率录制需要更长时间。 --- ### **3. 排查步骤** #### **步骤1:检查设备支持能力** - 通过 `CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES` 确认设备支持 `REQUEST_AVAILABLE_CAPABILITIES_CONSTRAINED_HIGH_SPEED_VIDEO`。 - 使用 `CameraCharacteristics.getHighSpeedVideoSizes()` 和 `getHighSpeedVideoFpsRangesFor()` 检查支持的高帧率配置。 #### **步骤2:验证测试参数** - 确保测试中使用的分辨率、帧率、格式设备支持的配置完全匹配。 - 示例代码: ```java Size[] highSpeedSizes = characteristics.getHighSpeedVideoSizes(); Range<Integer>[] fpsRanges = characteristics.getHighSpeedVideoFpsRangesFor(size); ``` #### **步骤3:检查会话配置** - 确认 `CameraCaptureSession` 使用 `createConstrainedHighSpeedCaptureSession()` 方法创建。 - 确保输出Surface配置正确(如使用 `SurfaceTexture` 或 `MediaRecorder`)。 #### **步骤4:查看系统日志** - 通过 `adb logcat` 过滤相机相关日志(如 `tag:Camera`、`tag:CameraService`),寻找以下关键信息: - 错误码(如 `ERROR_CAMERA_IN_USE`)。 - 高帧率配置被拒绝的警告。 #### **步骤5:排除资源冲突** - 关闭其他可能占用相机的应用。 - 确保测试前释放相机资源(如调用 `CameraDevice.close()`)。 #### **步骤6:调整超时时间(临时测试)** - 修改测试代码中的超时时间(如从3000ms改为5000ms),观察是否因设备启动慢导致失败。 --- ### **4. 解决方案** #### **A. 设备不支持高帧率** - 若硬件不支持,需豁免此测试(需CTS认证例外)。 - 在设备的 `android.hardware.camera2.xml` 中声明不支持此功能。 #### **B. 配置错误** - 根据设备支持的 `HighSpeedVideoSizes` 和 `FpsRanges` 调整测试参数。 - 示例: ```java // 使用设备支持的具体配置 Size targetSize = new Size(1280, 720); Range<Integer> targetFps = new Range<>(120, 120); ``` #### **C. 驱动/固件问题** - 联系芯片厂商检查相机驱动逻辑,确保高帧率请求能正确触发 `CaptureStart` 事件。 - 更新相机HAL层(Hardware Abstraction Layer)。 #### **D. 优化资源管理** - 在测试前强制释放相机资源: ```java cameraManager.setTorchMode(cameraId, false); // 关闭闪光灯占用 ``` --- ### **5. 关键代码逻辑验证** 检查 `startSlowMotionRecording()` 中是否正确处理了高帧率会话的生命周期: ```java // 正确创建高速会话 CameraCaptureSession.StateCallback callback = ...; device.createConstrainedHighSpeedCaptureSession(outputs, callback, handler); // 提交高速请求 CaptureRequest request = session.getHighSpeedRequestBuilder(targetSurface) .set(CaptureRequest.CONTROL_AE_TARGET_FPS_RANGE, targetFps) .build(); session.setRepeatingRequest(request, ...); ``` --- ### **6. 参考资料** - Android官方文档:[Constrained High-Speed Video](https://developer.android.com/reference/android/hardware/camera2/CameraDevice#createConstrainedHighSpeedCaptureSession(java.util.List%3Candroid.view.Surface%3E,%20android.hardware.camera2.CameraCaptureSession.StateCallback,%20android.os.Handler)) - CTS测试源码:[RecordingTest.java](https://android.googlesource.com/platform/cts/+/master/tests/camera/src/android/hardware/camera2/cts/RecordingTest.java) --- 通过以上步骤,可定位并修复 `testConstrainedHighSpeedRecording` 的超时问题。若需进一步分析,需结合具体设备的日志和硬件能力。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

快乐崇拜234

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值