AudioRecord 录制播放PCM音频
public AudioRecord(int audioSource, int sampleRateInHz, int channelConfig, int audioFormat,
int bufferSizeInBytes)
- 构造AudioRecord 对象
- startRecording() 开始采集
- 在线程中将采集数据写入pcm文件
- stop() 停止采集
public void record(){
mAudioRecord.startRecording();
new Thread(new Runnable() {
@Override
public void run() {
FileOutputStream fileOutputStream=new FileOutputStream(recordFile);
byte[] buffer=new byte[mBufferSizeInBytes];
while(isRecording){
int read=mAudioRecord.read(buffer,0,mBufferSizeInBytes);
if(AudioRecord.ERROR_INVALID_OPERATION!=read){
fileOutputStream.write(buffer);
}
}
fileOutputStream.close();
}
}).start(); }
public void stopRecord(){
isRecording=false;
if(mAudioRecord.getState()==AudioRecord.RECORDSTATE_RECORDING){
mAudioRecord.stop();
}
mAudioRecord.release();
}使用AudioTrack播放- 构造AudioTrack
- play()
- 在线程中write() 写入pcm文件流
- release()回收资源
与MediaPlayer的区别MediaPlayer可以播放多种格式的声音文件,例如MP3,AAC,WAV,OGG,MIDI等。MediaPlayer会在framework层创建对应的音频解码器。而AudioTrack只能播放已经解码的PCM流,如果对比支持的文件格式的话则是AudioTrack只支持wav格式的音频文件,因为wav格式的音频文件大部分都是PCM流。AudioTrack不创建解码器,只能播放不需要解码的wav文件构造函数public AudioTrack(AudioAttributes attributes, AudioFormat format, int bufferSizeInBytes,
int mode, int sessionId)mAudioTrack.play();
new Thread(new Runnable(){
@Override
public void run() { try {
FileInputStream fileInputStream=new FileInputStream(filePath);
byte[] tempBuffer=new byte[mWriteMinBufferSize];
while (fileInputStream.available()>0){
int readCount= fileInputStream.read(tempBuffer);
if (readCount== AudioTrack.ERROR_INVALID_OPERATION||readCount==AudioTrack.ERROR_BAD_VALUE){
continue;
}
if (readCount!=0&&readCount!=-1){
mAudioTrack.write(tempBuffer,0,readCount);
}
}
Log.e("TAG","end");
} catch (Exception e) {
e.printStackTrace();
} }
}).start();实例:使用AudioRecord 采集PCM,AudioTrack 播放
package com.rexkell.mediaapplication; import android.media.AudioAttributes;
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.media.MediaRecorder;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button; import com.rexkell.mediaapplication.media.MediaConfig; import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.concurrent.Executors; /**
* author: rexkell
* date: 2019/7/22
* explain:
*/
public class AudioRecordActivity extends AppCompatActivity {
boolean isRecording=false;
//音频源 MIC指的是麦克风
private final int mAudioSource= MediaRecorder.AudioSource.MIC;
//(MediaRecoder 的采样率通常是8000Hz AAC的通常是44100Hz。 设置采样率为44100,目前为常用的采样率,官方文档表示这个值可以兼容所有的设置)
private final int SampleRateInHz=44100; //采样率
//输入声道
private final int channelInMono= AudioFormat.CHANNEL_CONFIGURATION_MONO;
private final int channelOutMono=AudioFormat.CHANNEL_OUT_MONO;
//指定音频量化位数 ,在AudioFormaat类中指定了以下各种可能的常量。通常我们选择ENCODING_PCM_16BIT和ENCODING_PCM_8BIT PCM代表的是脉冲编码调制,它实际上是原始音频样本。
//因此可以设置每个样本的分辨率为16位或者8位,16位将占用更多的空间和处理能力,表示的音频也更加接近真实
private final int mAudioFormat=AudioFormat.ENCODING_PCM_16BIT;
//指定缓冲区大小
private int mBufferSizeInBytes;
private String filePath; private int mWriteMinBufferSize;
private AudioAttributes mAudioAttributes;
AudioRecord mAudioRecord;
AudioTrack mAudioTrack;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.ac_audio_record);
filePath=getExternalFilesDir("Music").getPath()+"/test.pcm";
initAudioRecord();
}
private void initAudioRecord(){
mBufferSizeInBytes=AudioRecord.getMinBufferSize(SampleRateInHz,channelInMono,mAudioFormat);
mWriteMinBufferSize=AudioTrack.getMinBufferSize(SampleRateInHz,channelInMono,mAudioFormat);
mAudioRecord=new AudioRecord(mAudioSource,SampleRateInHz,channelInMono,mAudioFormat,mBufferSizeInBytes); AudioFormat audioFormat=new AudioFormat.Builder().setSampleRate(SampleRateInHz).setEncoding(mAudioFormat).setChannelMask(channelOutMono).build();
mAudioAttributes=new AudioAttributes.Builder().setUsage(AudioAttributes.USAGE_MEDIA).setContentType(AudioAttributes.CONTENT_TYPE_MUSIC).build();
mAudioTrack=new AudioTrack(mAudioAttributes,audioFormat,mWriteMinBufferSize, AudioTrack.MODE_STREAM, AudioManager.AUDIO_SESSION_ID_GENERATE);
File recordFile=new File(filePath);
if (!recordFile.exists()){
try {
recordFile.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
} }
public void record(View view){
isRecording=!isRecording;
String text=isRecording?"结束":"录制";
((Button)view).setText(text);
if (isRecording){
final File recordFile=new File(filePath);
if (recordFile.exists()){
recordFile.delete();
try {
recordFile.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
}
mAudioRecord.startRecording();
Executors.newSingleThreadExecutor().execute(new Runnable() {
@Override
public void run() {
try {
FileOutputStream fileOutputStream=new FileOutputStream(recordFile);
byte[] buffer=new byte[mBufferSizeInBytes];
while (isRecording){
int read=mAudioRecord.read(buffer,0,mBufferSizeInBytes);
if (AudioRecord.ERROR_INVALID_OPERATION!=read){
fileOutputStream.write(buffer);
}
}
fileOutputStream.close();
} catch (Exception e) {
e.printStackTrace();
}
}
});
}else {
if (mAudioRecord!=null&&isRecording){
if (mAudioRecord.getState()==AudioRecord.RECORDSTATE_RECORDING){
mAudioRecord.stop();
}
mAudioRecord.release();
isRecording=false;
} } }
public void play(View view){
mAudioTrack.play();
Executors.newSingleThreadExecutor().execute(new Runnable() {
@Override
public void run() {
try {
FileInputStream fileInputStream=new FileInputStream(filePath);
byte[] tempBuffer=new byte[mWriteMinBufferSize];
while (fileInputStream.available()>0){
int readCount= fileInputStream.read(tempBuffer);
if (readCount== AudioTrack.ERROR_INVALID_OPERATION||readCount==AudioTrack.ERROR_BAD_VALUE){
continue;
}
if (readCount!=0&&readCount!=-1){
mAudioTrack.write(tempBuffer,0,readCount);
}
}
Log.e("TAG","end");
} catch (Exception e) {
e.printStackTrace();
}
}
});
} @Override
protected void onDestroy() {
if (mAudioRecord!=null){
mAudioRecord.release();
mAudioRecord=null;
}
if (mAudioTrack!=null){
mAudioTrack.release();
mAudioTrack=null;
}
super.onDestroy();
}
}
AudioRecord 录制播放PCM音频的更多相关文章
- 使用AudioTrack播放PCM音频数据(android)
众所周知,Android的MediaPlayer包含了Audio和video的播放功能,在Android的界面上,Music和Video两个应用程序都是调用MediaPlayer实现的.MediaPl ...
- 使用WindowsAPI实现播放PCM音频的方法
这篇文章主要介绍了使用WindowsAPI实现播放PCM音频的方法,很实用的一个功能,需要的朋友可以参考下 本文介绍了使用WindowsAPI实现播放PCM音频的方法,同前面一篇使用WindowsAP ...
- linux下mono播放PCM音频
测试环境: Ubuntu 14 MonoDevelop CodeBlocks 1.建立一个共享库(shared library) 这里用到了linux下的音频播放库,alsa-lib. al ...
- Android 音视频开发(三):使用 AudioTrack 播放PCM音频
一.AudioTrack 基本使用 AudioTrack 类可以完成Android平台上音频数据的输出任务.AudioTrack有两种数据加载模式(MODE_STREAM和MODE_STATIC),对 ...
- 使用WindowsAPI播放PCM音频
这一篇文章同上一篇<使用WindowsAPI获取录音音频>原理具有相似之处,不再详细介绍函数与结构体的参数 1. waveOutGetNumDevs 2. waveOutGetDevCap ...
- 11.3、Libgdx的音频之播放PCM音频
(官网:www.libgdx.cn) audio模块可以提供对音频硬件的直接访问. 音频硬件是通过AudioDevice接口进行的抽象. 以下创建一个新的AudioDevice实例: AudioDev ...
- DirectSound播放PCM(可播放实时采集的音频数据)
前言 该篇整理的原始来源为http://blog.csdn.net/leixiaohua1020/article/details/40540147.非常感谢该博主的无私奉献,写了不少关于不同多媒体库的 ...
- 最简单的视音频播放示例9:SDL2播放PCM
本文记录SDL播放音频的技术.在这里使用的版本是SDL2.实际上SDL本身并不提供视音频播放的功能,它只是封装了视音频播放的底层API.在Windows平台下,SDL封装了Direct3D这类的API ...
- 最简单的视音频播放示例8:DirectSound播放PCM
本文记录DirectSound播放音频的技术.DirectSound是Windows下最常见的音频播放技术.目前大部分的音频播放应用都是通过DirectSound来播放的.本文记录一个使用Direct ...
随机推荐
- 如何减小ABAP业务代码的复杂度
在程序开发的过程中,相同的功能往往有不同的实现方式.对于可以实现同样功能的不同代码,复杂度是用于比较其质量优劣的重要指标. 在本文中,代码复杂度是指代码被理解/修改的难易程度.越容易被理解.修改的代码 ...
- jimdb压测踩坑记
本文记录在jimdb压测过程中遇到的各种小坑,望能够抛砖引玉. 1.压测流量起来后,过了5分钟左右,发现ops突降,大概降了三分之一,然后稳定了下来 大概原因:此种情况,jimdb极有可能某个分片的连 ...
- ConcurrentHashMap源码走读
目录 ConcurrentHashMap源码走读 简介 放入数据 容器元素总数更新 容器扩容 协助扩容 遍历 ConcurrentHashMap源码走读 简介 在从JDK8开始,为了提高并发度,Con ...
- ReactNative: 使用弹出框组件ActionSheetIOS组件
一.简介 在上一篇文章中,详细介绍了对话框组件,除此之外,React-Native系统还给开发者提供了一个弹框框组件ActionSheetIOS组件,它的弹出方式与对框框不同,它是从底部弹出,与iOS ...
- IT兄弟连 HTML5教程 CSS3属性特效 倒影
在Web制作中,有些时候需要实现一些倒影的效果.在传统网页中,我们只能使用photoshop事先将倒影设计好,然后导入到网页中,这样不但耗费资源,也阻碍了开发效率.而CSS新增了Reflections ...
- 解决Entity 实体类中加了@Id 注解后仍然出现org.hibernate.AnnotationException: No identifier specified for entity 错误
启动报错如下图所示: 解决方案: 查看网上的资料,大部分都说在实体类中没有添加加主键的注解@Id,这个是必须的.但是我的实体类中明明已经添加了@Id,为什么还会报这个错误呢? 后来检查了很久,发现是我 ...
- ASP.NET Core gRPC 健康检查的实现方式
一. 前言 gRPC 服务实现健康检查有两种方式,前面在此文 ASP.NET Core gRPC 使用 Consul 服务注册发现 中有提到过,这里归纳整理一下.gRPC 的健康检查,官方是定义了标准 ...
- 流式计算(二)-Kafka Stream
前面说了Java8的流,这里还说流处理,既然是流,比如水流车流,肯定得有流的源头,源可以有多种,可以自建,也可以从应用端获取,今天就拿非常经典的Kafka做源头来说事,比如要来一套应用日志实时分析框架 ...
- Pyhton中变量和数据类型
一.变量 Python中变量的命名规则: 1.变量名只能包含数字.字母.下划线,且不能用数字打头. eg: message_1是对的但1_message就是错误的 2.变量名不能包含空格. 3.在变量 ...
- 5.Ansible Jinja2 模板
1.jinja2渲染NginxProxy配置文件 jinja2 房屋建筑设计固定的 jinja2模板与Ansible关系 Ansible如何使用jinja2模板 template模块 拷贝文件? te ...