MediaPlayer和AudioTrack播放Audio的区别与联系

2024-04-22 17:32

本文主要是介绍MediaPlayer和AudioTrack播放Audio的区别与联系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

播放声音可以用MediaPlayer和AudioTrack,两者都提供了java API供应用开发者使用。虽然都可以播放声音,但两者还是有很大的区别的。
其中最大的区别是MediaPlayer可以播放多种格式的声音文件,例如MP3,AAC,WAV,OGG,MIDI等。MediaPlayer会在framework层创建对应的音频解码器。
而AudioTrack只能播放已经解码的PCM流,如果是文件的话只支持wav格式的音频文件,因为wav格式的音频文件大部分都是PCM流。AudioTrack不创建解码器,所以只能播放不需要解码的wav文件。

当然两者之间还是有紧密的联系的,MediaPlayer在framework层还是会创建AudioTrack,把解码后的PCM数流传递给AudioTrack,AudioTrack再传递给AudioFlinger进行混音,然后才传递给硬件播放。
所以是MediaPlayer包含了AudioTRack。
通过查看API可以知道,MediaPlayer提供了5个setDataSource方法,分为三类,一类是传递播放文件的字符串路径作为参数,例如直接取sd卡里mp3文件的路径,一类是传递播放文件的FileDescriptor文件描述符作为播放的id,例例如从db中查询的音频文件的id,就可以直接赋给MediaPlayer进行播放。还有一类是Uri类型的资源文件,用于播放content uri文件。

下面是一个用MediaPlayer播放音乐的示例,音乐文件是从数据库中取出的。
[java] view plain copy
  1. if (mMediaPlayer == null) {  
  2.     mMediaPlayer = new MediaPlayer(); // 创建MediaPlayer对象  
  3. }  
  4.   
  5. mMediaPlayer.reset();  
  6. String dataSource = getDataByPosition(mCursor, mPlayPosition);  
  7. mMediaPlayer.setDataSource(dataSource); // 设置需要播放的数据源  
  8. mMediaPlayer.prepare(); // 准备播放,如果是流媒体需要调用prepareAsync进行异步准备  
  9. if (Common.PLAY_MODE_SINGLE_LOOP == mPlayMode) {  
  10.     mMediaPlayer.setLooping(true); // 单曲循环  
  11. else {  
  12.     mMediaPlayer.setLooping(false); // 不循环播放  
  13. }  
  14. mMediaPlayer.start(); // 开始播放,如果是播放流媒体,需要等到流媒体准备完成才能播放(在prepare的callback函数中调用start进行播放)  
  15.   
  16. // 根据位置来获取歌曲位置  
  17. public String getDataByPosition(Cursor c, int position) {  
  18.     c.moveToPosition(position);  
  19.     int dataColumn = c.getColumnIndex(MediaStore.Audio.Media.DATA);  
  20.     String data = c.getString(dataColumn);  
  21.     return data;  
  22. }  
AudioTrack播放声音时不能直接把wav文件传递给AudioTrack进行播放,必须传递buffer,通过write函数把需要播放的缓冲区buffer传递给AudioTrack,然后才能播放。

AudioTrack使用的例子参考下面:
此示例转自:http://samyou.iteye.com/blog/1125872
[java] view plain copy
  1. public class AndroidTest extends Activity implements View.OnClickListener,SurfaceHolder.Callback  
  2. {  
  3.     private SurfaceHolder surfaceHolder = null;  
  4.     private SurfaceView surfaceView = null;  
  5.     private AudioTrack audioTrack = null;  
  6.     private Thread writePCMThread = null;  
  7.     private File audioFile = null;  
  8.     private FileInputStream fileInputStream = null;  
  9.     private byte buffer[] = new byte[16*10000];  
  10.   
  11.     // The Handler that gets information back from the other threads  
  12.     private final Handler msgHandler = new Handler()  
  13.     {  
  14.         public void handleMessage(Message msg)  
  15.         {  
  16.             switch (msg.what)  
  17.             {  
  18.             default:  
  19.                 break;  
  20.             }  
  21.         }  
  22.     };  
  23.   
  24.   
  25.     /** Called when the activity is first created. */  
  26.     @Override  
  27.     public void onCreate(Bundle savedInstanceState) {  
  28.         super.onCreate(savedInstanceState);  
  29.         setContentView(R.layout.main);  
  30.         surfaceView = (SurfaceView) findViewById(R.id.surface);  
  31.         SurfaceHolder surfaceHolder = surfaceView.getHolder();  
  32.         surfaceHolder.addCallback(this);  
  33.         surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);  
  34.         findViewById(R.id.button1).setOnClickListener(this);  
  35.         findViewById(R.id.button2).setOnClickListener(this);  
  36.     }  
  37.   
  38.     public void finish() {  
  39.         super.finish();  
  40.         System.out.println("finish");  
  41.         try {  
  42.             writePCMThread.interrupt();  
  43.         } catch (Exception e) {  
  44.         }  
  45.         try {  
  46.             fileInputStream.close();  
  47.         } catch (Exception e) {  
  48.         }  
  49.         try {  
  50.             audioTrack.stop();  
  51.             audioTrack.release();  
  52.         } catch (Exception e) {  
  53.         }  
  54.     }  
  55.   
  56.   
  57.   
  58.     protected void onResume()  
  59.     {  
  60.         super.onResume();  
  61.         System.out.println("back on!!!!!!!!!!!");  
  62.         initAudioTrack();  
  63.         audioFile = new File(Environment.getExternalStorageDirectory(),"test.wav");  
  64.         System.out.println(audioFile.length());  
  65.         try {  
  66.             fileInputStream = new FileInputStream(audioFile);  
  67.             fileInputStream.skip(0x2c);  
  68.         } catch (Exception e) {  
  69.         }  
  70.   
  71.         writePCMThread = new Thread(new Runnable(){  
  72.             public void run() {  
  73.                 try  
  74.                 {  
  75.                     while(fileInputStream.read(buffer)>=0)  
  76.                     {  
  77.                         System.out.println("write pcm data");  
  78.                         audioTrack.write(buffer, 0, buffer.length);  
  79.                     }  
  80.                 }  
  81.                 catch (Exception e) {  
  82.                     e.printStackTrace();  
  83.                 }  
  84.             }  
  85.         });  
  86.   
  87.     }  
  88.   
  89.     private void initAudioTrack()  
  90.     {  
  91.         int minBufferSize = AudioTrack.getMinBufferSize(0xac44, AudioFormat.CHANNEL_CONFIGURATION_MONO, AudioFormat.ENCODING_PCM_16BIT);  
  92.         System.out.println("minBufferSize = "+minBufferSize);  
  93.         audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, 0xac44,  
  94.         AudioFormat.CHANNEL_CONFIGURATION_STEREO, AudioFormat.ENCODING_PCM_16BIT, minBufferSize*2,AudioTrack.MODE_STREAM);  
  95.         audioTrack.setStereoVolume(1.0f, 1.0f);// 设置当前音量大小  
  96.         System.out.println("initAudioTrack over");  
  97.         audioTrack.play();  
  98.     }  
  99.   
  100.     public void onClick(View v)  
  101.     {  
  102.         switch (v.getId()) {  
  103.         case R.id.button1:  
  104.             writePCMThread.start();  
  105.             break;  
  106.         case R.id.button2:  
  107.             break;  
  108.         default:  
  109.             break;  
  110.     }  
  111.   
  112.   
  113.     }  
  114.   
  115.     public void surfaceCreated(SurfaceHolder holder) {  
  116.         System.out.println("surfaceCreated()");  
  117.         this.surfaceHolder = holder;  
  118.     }  
  119.   
  120.     public void surfaceDestroyed(SurfaceHolder holder) {  
  121.     }  
  122.   
  123.     public void surfaceChanged(SurfaceHolder holder, int format, int width,  
  124.         int height) {  
  125.         this.surfaceHolder = holder;  
  126.     }  
  127.   
  128.     public void onActivityResult(int requestCode, int resultCode, Intent data) {  
  129.     }  
  130.   

   转至  : http://blog.csdn.net/ameyume/article/details/7618820

这篇关于MediaPlayer和AudioTrack播放Audio的区别与联系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/926407

相关文章

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

Python中@classmethod和@staticmethod的区别

《Python中@classmethod和@staticmethod的区别》本文主要介绍了Python中@classmethod和@staticmethod的区别,文中通过示例代码介绍的非常详细,对大... 目录1.@classmethod2.@staticmethod3.例子1.@classmethod

Golan中 new() 、 make() 和简短声明符的区别和使用

《Golan中new()、make()和简短声明符的区别和使用》Go语言中的new()、make()和简短声明符的区别和使用,new()用于分配内存并返回指针,make()用于初始化切片、映射... 详细介绍golang的new() 、 make() 和简短声明符的区别和使用。文章目录 `new()`

Python中json文件和jsonl文件的区别小结

《Python中json文件和jsonl文件的区别小结》本文主要介绍了JSON和JSONL两种文件格式的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下... 众所周知,jsON 文件是使用php JSON(JavaScripythonpt Object No

结构体和联合体的区别及说明

《结构体和联合体的区别及说明》文章主要介绍了C语言中的结构体和联合体,结构体是一种自定义的复合数据类型,可以包含多个成员,每个成员可以是不同的数据类型,联合体是一种特殊的数据结构,可以在内存中共享同一... 目录结构体和联合体的区别1. 结构体(Struct)2. 联合体(Union)3. 联合体与结构体的

什么是 Ubuntu LTS?Ubuntu LTS和普通版本区别对比

《什么是UbuntuLTS?UbuntuLTS和普通版本区别对比》UbuntuLTS是Ubuntu操作系统的一个特殊版本,旨在提供更长时间的支持和稳定性,与常规的Ubuntu版本相比,LTS版... 如果你正打算安装 Ubuntu 系统,可能会被「LTS 版本」和「普通版本」给搞得一头雾水吧?尤其是对于刚入

python中json.dumps和json.dump区别

《python中json.dumps和json.dump区别》json.dumps将Python对象序列化为JSON字符串,json.dump直接将Python对象序列化写入文件,本文就来介绍一下两个... 目录1、json.dumps和json.dump的区别2、使用 json.dumps() 然后写入文

流媒体平台/视频监控/安防视频汇聚EasyCVR播放暂停后视频画面黑屏是什么原因?

视频智能分析/视频监控/安防监控综合管理系统EasyCVR视频汇聚融合平台,是TSINGSEE青犀视频垂直深耕音视频流媒体技术、AI智能技术领域的杰出成果。该平台以其强大的视频处理、汇聚与融合能力,在构建全栈视频监控系统中展现出了独特的优势。视频监控管理系统EasyCVR平台内置了强大的视频解码、转码、压缩等技术,能够处理多种视频流格式,并以多种格式(RTMP、RTSP、HTTP-FLV、WebS

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

native和static native区别

本文基于Hello JNI  如有疑惑,请看之前几篇文章。 native 与 static native java中 public native String helloJni();public native static String helloJniStatic();1212 JNI中 JNIEXPORT jstring JNICALL Java_com_test_g