編輯:Android編程入門
公司產品有很多地方都需要上傳音頻視頻,今天抽空總結一下音頻視頻的錄制。學習的主角是MediaRecorder類。
MediaRecorder類是Android sdk提供的一個專門用於音視頻錄制,一般利用手機麥克風采集音頻,攝像頭采集圖片信息。
setAudioChannels(int numChannels) 設置錄制的音頻通道數
setAudioEncoder(int audio_encoder) 設置audio的編碼格式
setAudioEncodingBitRate(int bitRate) 設置錄制的音頻編碼比特率
setAudioSamplingRate(int samplingRate) 設置錄制的音頻采樣率
setAudioSource(int audio_source) 設置用於錄制的音源
setAuxiliaryOutputFile(String path) 輔助時間的推移視頻文件的路徑傳遞
setAuxiliaryOutputFile(FileDescriptor fd)在文件描述符傳遞的輔助時間的推移視頻
setCamera(Camera c) 設置一個recording的攝像頭
setCaptureRate(double fps) 設置視頻幀的捕獲率
setMaxDuration(int max_duration_ms) 設置記錄會話的最大持續時間(毫秒)
setMaxFileSize(long max_filesize_bytes) 設置記錄會話的最大大小(以字節為單位)
setOutputFile(FileDescriptor fd) 傳遞要寫入的文件的文件描述符
setOutputFile(String path) 設置輸出文件的路徑
setOutputFormat(int output_format) 設置在錄制過程中產生的輸出文件的格式
setPreviewDisplay(Surface sv) 表面設置顯示記錄媒體(視頻)的預覽
setVideoEncoder(int video_encoder) 設置視頻編碼器,用於錄制
setVideoEncodingBitRate(int bitRate) 設置錄制的視頻編碼比特率
setVideoFrameRate(int rate) 設置要捕獲的視頻幀速率
setVideoSize(int width, int height) 設置要捕獲的視頻的寬度和高度
setVideoSource(int video_source) 開始捕捉和編碼數據到setOutputFile(指定的文件)
setLocation(float latitude, float longitude) 設置並存儲在輸出文件中的地理數據(經度和緯度)
setProfile(CamcorderProfile profile) 指定CamcorderProfile對象
setOrientationHint(int degrees)設置輸出的視頻播放的方向提示
setOnErrorListener(MediaRecorder.OnErrorListener l)注冊一個用於記錄錄制時出現的錯誤的監聽器
setOnInfoListener(MediaRecorder.OnInfoListener listener)注冊一個用於記錄錄制時出現的信息事件
getMaxAmplitude() 獲取在前一次調用此方法之後錄音中出現的最大振幅
prepare()准備錄制。
release()釋放資源
reset()將MediaRecorder設為空閒狀態
start()開始錄制
stop()停止錄制
default,H263,H264,MPEG_4_SP,VP8
default,AAC,HE_AAC,AAC_ELD,AMR_NB,AMR_WB,VORBIS
default,CAMERA,SURFACE
defalut,camcorder,mic,voice_call,voice_communication,voice_downlink,voice_recognition, voice_uplink
amr_nb,amr_wb,default,mpeg_4,raw_amr,three_gpp,aac_adif, aac_adts, output_format_rtp_avp, output_format_mpeg2ts ,webm
<uses-permission android:name="android.permission.CAMERA" /> <uses-permission android:name="android.permission.RECORD_AUDIO" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
public class MediaRecorderActivity extends Activity implements SurfaceHolder.Callback { private static final String TAG = MediaRecorderActivity.class.getSimpleName(); private SurfaceView mSurfaceView; private Button btnStartStop; private boolean isStart = false; private MediaRecorder mRecorder; private SurfaceHolder mSurfaceHolder; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setWindow(); setContentView(R.layout.activity_media_recorder); initViews(); } private void setWindow() { requestWindowFeature(Window.FEATURE_NO_TITLE);// 去掉標題欄 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);// 設置全屏 // 設置豎屏顯示 setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT); // 選擇支持半透明模式,在有surfaceview的activity中使用。 getWindow().setFormat(PixelFormat.TRANSLUCENT); } private void initViews() { mSurfaceView = (SurfaceView) findViewById(R.id.surfaceview); btnStartStop = (Button) findViewById(R.id.btnStartStop); btnStartStop.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { startRecord(); } }); SurfaceHolder holder = mSurfaceView.getHolder();// 取得holder holder.setFormat(PixelFormat.TRANSPARENT); holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); holder.setKeepScreenOn(true); holder.addCallback(this); // holder加入回調接口 } private void startRecord() { if (!isStart) { // 開始錄制 if (mRecorder == null) { mRecorder = new MediaRecorder(); // 創建MediaRecorder } try { // 設置音頻采集方式 mRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER); //設置視頻的采集方式 mRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA); //設置文件的輸出格式 // mRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);//aac_adif, aac_adts, output_format_rtp_avp, output_format_mpeg2ts ,webm //設置audio的編碼格式 mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB); //設置video的編碼格式 mRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264); //設置錄制的視頻編碼比特率 mRecorder.setVideoEncodingBitRate(1024 * 1024); //設置錄制的視頻幀率,注意文檔的說明: mRecorder.setVideoFrameRate(30); //設置要捕獲的視頻的寬度和高度 mSurfaceHolder.setFixedSize(320, 240);//最高只能設置640x480 mRecorder.setVideoSize(320, 240);//最高只能設置640x480 //設置記錄會話的最大持續時間(毫秒) mRecorder.setMaxDuration(60 * 1000); mRecorder.setPreviewDisplay(mSurfaceHolder.getSurface()); String path = getExternalCacheDir().getPath(); if (path != null) { File dir = new File(path + "/videos"); if (!dir.exists()) { dir.mkdir(); } path = dir + "/" + System.currentTimeMillis() + ".mp4"; //設置輸出文件的路徑 mRecorder.setOutputFile(path); //准備錄制 mRecorder.prepare(); //開始錄制 mRecorder.start(); isStart = true; btnStartStop.setText("停止"); } } catch (Exception e) { e.printStackTrace(); } } else { } } private void stopRecord(){ if (isStart) { try { //停止錄制 mRecorder.stop(); //重置 mRecorder.reset(); btnStartStop.setText("開始"); } catch (Exception e) { e.printStackTrace(); } } isStart = false; } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { // 將holder,這個holder為開始在onCreate裡面取得的holder,將它賦給mSurfaceHolder mSurfaceHolder = holder; } @Override public void surfaceCreated(SurfaceHolder holder) { // 將holder,這個holder為開始在onCreate裡面取得的holder,將它賦給mSurfaceHolder mSurfaceHolder = holder; } @Override public void surfaceDestroyed(SurfaceHolder holder) { // surfaceDestroyed的時候同時對象設置為null mSurfaceView = null; mSurfaceHolder = null; if (mRecorder != null) { mRecorder.release(); // Now the object cannot be reused mRecorder = null; } } }
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <SurfaceView android:id="@+id/surfaceview" android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1" /> <Button android:id="@+id/btnStartStop" android:layout_width="wrap_content" android:layout_height="55dip" android:layout_gravity="center" android:text="開始" tools:context=".MainActivity" /> </LinearLayout>
自從Google在2013年發布了Android Studio後,Android Studio憑借著自己良好的內存優化,酷炫的UI主題,強大的自動補全提示以及Gradle
讓我們來簡單了解下Android Studio中不同目錄(文件)的位置和用途。首先看下一個App的最簡單的目錄結構 OK,我們這麼看,第一,把這麼多
一,簡述線程池:線程池是如何工作的:一系列任務出現後,根據自己的線程池安排任務進行。如圖: 線程池的好處:重用線程池中的線程,避免因為線程的創建和銷毀所帶來的性能開銷。能
如果說Android上的app是一個有血有肉的人的話,那麼人靠衣裝馬靠鞍,那麼控件就是把app裝扮的漂漂亮亮的“衣服”。那麼安卓的控件到底是如何架