Android多媒体开发【6】-- 播放器的基本模型

AwesomePlayer 不管他有多神秘,说到底还是个多媒体播放器。在播放器的基本模型上,他与VCL、mplayer、ffmpeg等开源的结构是一致的。只是组织实现的方式不同。

深入了解AwesomePlayer 之前,把播放器的基本模型总结一下,然后按照模型的各个部分来深入研究AwesomePlayer 的实现方式。


说白了播放器大致分为4大部分:source、demux、decoder、output。

1.source:数据源,数据的来源不一定都是本地file,也有可能是网路上的各种协议例如:http、rtsp、HLS等。source的任务就是把数据源抽象出来,为下一个demux模块提供它需要的稳定的数据流。demux不用关信数据到底是从什么地方来的。

2.demux解复用:视频文件一般情况下都是把音视频的ES流交织的通过某种规则放在一起。这种规则就是容器规则。现在有很多不同的容器格式。如ts、mp4、flv、mkv、avi、rmvb等等。demux的功能就是把音视频的ES流从容器中剥离出来,然后分别送到不同的解码器中。其实音频和视频本身就是2个独立的系统。容器把它们包在了一起。但是他们都是独立解码的,所以解码之前,需要把它分别 独立出来。demux就是干这活的,他为下一步decoder解码提供了数据流。

3.decoder解码:解码器--播放器的核心模块。分为音频和视频解码器。影像在录制后, 原始的音视频都是占用大量空间, 而且是冗余度较高的数据. 因此, 通常会在制作的时候就会进行某种压缩 ( 压缩技术就是将数据中的冗余信息去除数据之间的相关性 ). 这就是我们熟知的音视频编码格式, 包括MPEG1(VCD)\ MPEG2(DVD)\ MPEG4 \ H.264 等等. 音视频解码器的作用就是把这些压缩了的数据还原成原始的音视频数据. 当然, 编码解码过程基本上都是有损的 .解码器的作用就是把编码后的数据还原成原始数据。

4.output输出:输出部分分为音频和视频输出。解码后的音频(pcm)和视频(yuv)的原始数据需要得到音视频的output模块的支持才能真正的让人的感官系统(眼和耳)辨识到。

所以,播放器大致分成上述4部分。怎么抽象的实现这4大部分、以及找到一种合理的方式将这几部分组织并运动起来。是每个播放器不同的实现方式而已。

接下来就围绕这4大部分做深入学习,看看AwesomePlayer 是怎么玩的吧。


  • 7
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
package com.demo.pr5; import java.io.File; import java.util.Vector; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.KeyEvent; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.BaseAdapter; import android.widget.Button; import android.widget.EditText; import android.widget.ImageView; import android.widget.ListView; import android.widget.TextView; import android.widget.Toast; import android.widget.AdapterView.OnItemClickListener; public class MyFileActivity extends Activity { // 支持的媒体格式 private final String[]FILE_MapTable = { ".3gp",".mov",".avi", ".rmvb", ".wmv", ".mp3", ".mp4" }; private Vector<String> items = null; // items:存放显示的名称 private Vector<String> paths = null; // paths:存放文件路径 private Vector<String> sizes = null; // sizes:文件大小 private String rootPath = "/mnt/sdcard"; //起始文件夹 private EditText pathEditText; // 路径 private Button queryButton; //查询按钮 private ListView fileListView;//文件列表 @Override protected void onCreate(Bundle icicle) { super.onCreate(icicle); this.setTitle("多媒体文件浏览"); setContentView(R.layout.myfile); //从myfile.xml找到对应的元素 pathEditText = (EditText) findViewById(R.id.path_edit); queryButton = (Button) findViewById(R.id.qry_button); fileListView= (ListView) findViewById(R.id.file_listview); //查询按钮事件 queryButton.setOnClickListener( new Button.OnClickListener() { public void onClick(View arg0) { File file = new File(pathEditText.getText().toString()); if (file.exists()) { if (file.isFile()) { //如果是媒体文件直接打开播放 openFile(pathEditText.getText().toString()); } else { //如果是目录打开目录下文件 getFileDir(pathEditText.getText().toString()); } } else { Toast.makeText(MyFileActivity.this, "找不到该位置,请确定位置是否正确!", Toast.LENGTH_SHORT).show(); } } }); //设置ListItem被点击时要做的动作 fileListView.setOnItemClickListener(new OnItemClickListener() { @Override public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) { fileOrDir(paths.get(position)); } }); //打开默认文件夹 getFileDir(rootPath); } /** * 重写返回键功能:返回上一级文件夹 */ @Override public boolean onKeyDown(int keyCode, KeyEvent event) { // 是否触发按键为back键 if (keyCode == KeyEvent.KEYCODE_BACK) { pathEditText = (EditText) findViewById(R.id.path_edit); File file = new File(pathEditText.getText().toString()); if (rootPath.equals(pathEditText.getText().toString().trim())) { return super.onKeyDown(keyCode, event); } else { getFileDir(file.getParent()); return true; } //如果不是back键正常响应 } else { return super.onKeyDown(keyCode, event); } } /** * 处理文件或者目录的方法 */ private void fileOrDir(String path) { File file = new File(path); if (file.isDirectory()) { getFileDir(file.getPath()); } else { openFile(path); } } /** * 取得文件结构的方法 */ private void getFileDir(String filePath) { /* 设置目前所在路径 */ pathEditText.setText(filePath); items = new Vector<String>(); paths = new Vector<String>(); sizes = new Vector<String>(); File f = new File(filePath); File[] files = f.listFiles(); if (files != null) { /* 将所有文件添加ArrayList中 */ for (int i = 0; i < files.length; i++) { if (files[i].isDirectory()) { items.add(files[i].getName()); paths.add(files[i].getPath()); sizes.add(""); } } for (int i = 0; i < files.length; i++) { if (files[i].isFile()) { String fileName = files[i].getName(); int index = fileName.lastIndexOf("."); if (index > 0) { String endName = fileName.substring(index, fileName.length()).toLowerCase(); String type = null; for (int x = 0; x < FILE_MapTable.length; x++) { // 支持的格式,才会在文件浏览器中显示 if (endName.equals(FILE_MapTable[x])) { type = FILE_MapTable[x]; break; } } if (type != null) { items.add(files[i].getName()); paths.add(files[i].getPath()); sizes.add(files[i].length()+""); } } } } } /* 使用自定义的FileListAdapter来将数据传入ListView */ fileListView.setAdapter(new FileListAdapter(this, items)); } /** * 打开媒体文件 * @param f */ private void openFile(String path) { //打开媒体播放器 Intent intent = new Intent(MyFileActivity.this, MediaPlayerActivity.class); intent.putExtra("path",path); startActivity(intent); finish(); } /** *ListView列表适配器 */ class FileListAdapter extends BaseAdapter { private Vector<String> items = null; // items:存放显示的名称 private MyFileActivity myFile; public FileListAdapter(MyFileActivity myFile, Vector<String> items) { this.items=items; this.myFile=myFile; } @Override public int getCount() { // TODO Auto-generated method stub return items.size(); } @Override public Object getItem(int position) { // TODO Auto-generated method stub return items.elementAt(position); } @Override public long getItemId(int position) { // TODO Auto-generated method stub return items.size(); } @Override public View getView(int position, View convertView, ViewGroup parent) { // TODO Auto-generated method stub if(convertView==null) { //加载列表项布局file_item.xml convertView = myFile.getLayoutInflater() .inflate(R.layout.file_item, null); } //文件名称 TextView name = (TextView) convertView.findViewById(R.id.name); //媒体文件类型 ImageView music=(ImageView)convertView.findViewById(R.id.music); //文件夹类型 ImageView folder=(ImageView)convertView.findViewById(R.id.folder); name.setText(items.elementAt(position)); if(sizes.elementAt(position).equals("")) { //隐藏媒体图标,显示文件夹图标 music.setVisibility(View.GONE); folder.setVisibility(View.VISIBLE); }else { //隐藏文件夹图标,显示媒体图标 folder.setVisibility(View.GONE); music.setVisibility(View.VISIBLE); } return convertView; } } }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值