本文将引导大家做一个音乐播放器,在做这个Android开发实例的过程中,能够帮助大家进一步熟悉和掌握学过的ListView和其他一些组件。为了有更好的学习效果,其中很多功能我们手动实现,例如音乐播放的快进快退等。
先欣赏下本实例完成后运行的界面效果:
首先我们建立项目,我使用的SDK是Android2.2的,然后在XML中进行布局。
上方是一个ListView用来显示我们的音乐列表,中间是一个SeekBar可以拖动当前音乐的播放进度,之所以用SeekBar而不用ProgressBar是因为我们需要音乐的快进快退功能,可以拖动滑杆改变进度;还有一个TextView,用来显示当前播放歌曲的名字,时长等。最下方就是4个Button了,分别是上一曲,播放(暂停),停止,下一曲。
大家注意尽量不要在布局中出现直接显示在界面上的文字内容,我们把这些内容都放在res/values下的strings.xml中,然后分别引用它们,这样养成良好的习惯,界面与内容分离,方便调试和后期维护等。现在我们的界面如下:
然后我们把File Explorer打开,在eclipse的Window -- Show View -- Other --Android --File Explore。你也可以直接Alt+Shift+Q。
在mnt/sdcard下面,我们放个两三首歌曲,在虚拟机中暂不支持中文,导入有中文的文件会报错的。
接着我们创建一个类,做我们播放器的Service类,我就叫MusicService吧,在里面声明以下对象:
- public class MusicService {
- private static final File MUSIC_PATH = Environment
- .getExternalStorageDirectory();// 找到music存放的路径。
- public List<String> musicList;// 存放找到的所有mp3的绝对路径。
- public MediaPlayer player; // 定义多媒体对象
- public int songNum; // 当前播放的歌曲在List中的下标
- public String songName; // 当前播放的歌曲名
- }
然后我们去加载刚才添加的MP3文件吧,这里的方式多种多样,我随便写一个简单的了:
- class MusicFilter implements FilenameFilter {
- public boolean accept(File dir, String name) {
- return (name.endsWith(".mp3"));//返回当前目录所有以.mp3结尾的文件
- }
- }
在MusicService类的无参构造函数中实例化对象,并把这些MP3文件放到musicList中。
- public MusicService() {
- musicList = new ArrayList<String>();
- player = new MediaPlayer();
- if (MUSIC_PATH.listFiles(new MusicFilter()).length > 0) {
- for (File file : MUSIC_PATH.listFiles(new MusicFilter())) {
- musicList.add(file.getAbsolutePath());
- }
- }
- }
我们写个方法,来设置当前播放歌曲的名字:(个人觉得这方法比较笨,但暂时没想到别的办法)
- public void setPlayName(String dataSource) {
- File file = new File(dataSource);//假设为D:\\mm.mp3
- String name = file.getName();//name=mm.mp3
- int index = name.lastIndexOf(".");//找到最后一个.
- songName = name.substring(0, index);//截取为mm
- }
接下来就是我们Service类的基本方法了,也就是开始、暂停、停止、上一首和下一首。
我们分别使用声明的多媒体对象的start、pause、stop等方法可以完成。
- public void start() {
- try {
- player.reset(); //重置多媒体
- String dataSource = musicList.get(songNum);//得到当前播放音乐的路径
- setPlayName(dataSource);//截取歌名
- player.setDataSource(dataSource);//为多媒体对象设置播放路径
- player.prepare();//准备播放
- player.start();//开始播放
- //setOnCompletionListener 当当前多媒体对象播放完成时发生的事件
- player.setOnCompletionListener(new OnCompletionListener() {
- public void onCompletion(MediaPlayer arg0) {
- next();//如果当前歌曲播放完毕,自动播放下一首.
- }
- });
- } catch (Exception e) {
- Log.v("MusicService", e.getMessage());
- }
- }
- public void next() {
- songNum = songNum == musicList.size() - 1 ? 0 : songNum + 1;
- start();
- }
- public void last() {
- songNum = songNum == 0 ? musicList.size() - 1 : songNum - 1;
- start();
- }
- public void pause() {
- if (player.isPlaying())
- player.pause();
- else
- player.start();
- }
- public void stop() {
- if (player.isPlaying()) {
- player.stop();
- }
- }
到此为止我们的Service类就写完了,接着我们去Activity中为各控件绑定事件。
在这个Activity中,最难做的一点应该就是拖动SeekBar的滑杆改变播放进度了,这里我考虑再三,用了一个Handler类来处理。
Handler在android里负责发送和处理消息。它的主要用途有:
1.按计划发送消息或执行某个Runnanble(使用POST方法)。
2.从其他线程中发送来的消息放入消息队列中,避免线程冲突(常见于更新UI线程)。
默认情况下,Handler接受的是当前线程下的消息循环实例(使用Handler(Looper looper)、Handler(Looper looper, Handler.Callback callback)可以指定线程),同时一个消息队列可以被当前线程中的多个对象进行分发、处理(在UI线程中,系统已经有一个Activity来处理了,你可以再起若干个Handler来处理)。在实例化Handler的时候,Looper可以是任意线程的,只要有Handler的指针,任何线程也都可以sendMessage。Handler对于Message的处理不是并发的。一个Looper 只有处理完一条Message才会读取下一条,所以消息的处理是阻塞形式的(handleMessage()方法里不应该有耗时操作,可以将耗时操作放在其他线程执行,操作完后发送Message(通过sendMessges方法),然后由handleMessage()更新UI)。
声明以下变量:
- private Button btnStart, btnStop, btnNext, btnLast;
- private TextView txtInfo;
- private ListView listView;
- private SeekBar seekBar;
- private MusicService musicService;
- private MusicHandler musicHandler;// 处理改变进度条事件
- private MusicThread musicThread;// 自动改变进度条的线程
- private boolean autoChange, manulChange;// 判断是进度条是自动改变还是手动改变
- private boolean isPause;// 判断是从暂停中恢复还是重新播放
如有报错的可以先注释掉不用管它,然后在初始化过程中绑定事件。
这是ListView的填充方法:
- private void setListViewAdapter() {
- List<Map<String, Object>> date = new ArrayList<Map<String, Object>>();
- for (String path : musicService.musicList) {
- Map<String, Object> map = new HashMap<String, Object>();
- File file = new File(path);
- map.put("fileName", file.getName());
- date.add(map);
- }
- SimpleAdapter adapter = new SimpleAdapter(this, date,
- android.R.layout.simple_list_item_1,
- new String[] { "fileName" }, new int[] { android.R.id.text1 });
- listView.setAdapter(adapter);
- }
SimpleAdapter的构造函数是:
public SimpleAdapter (Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to);
第一个参数context,是指在哪个Activity中显示。
第二个参数是一个泛型作为数据源,而且每一个List中的一行就代表着呈现出来的一行,Map的键就是这一行的列名,值也是有列名的。
第三个参数为资源文件,就是说要加载这个列所需要的视图资源文件,我直接引用系统内置的资源,如果你想要漂亮的样式可以自己写的。
第四个参数是一个String数组,主要是将Map对象中的名称映射到列名,一一对应。
第五个是将第四个参数的值一一对象的显示(一一对应)在接下来的int形的id数组中,这个id数组就是Layout的xml文件中命名id形成的唯一的int型标识符。
SeekBar停止拖动后的事件:
- public void onStopTrackingTouch(SeekBar seekBar) { // 停止拖动
- int progress = seekBar.getProgress();
- if (!autoChange && manulChange) {
- int musicMax = musicService.player.getDuration(); //得到该首歌曲最长秒数
- int seekBarMax = seekBar.getMax();
- musicService.player
- .seekTo(musicMax * progress / seekBarMax);//跳到该曲该秒
- musicService.pause();
- autoChange = true;
- manulChange = false;
- }
- }
MusicHandler类的实现:
- class MusicHandler extends Handler {
- public MusicHandler() {
- }
- @Override
- public void handleMessage(Message msg) {
- if (autoChange) {
- try {
- int position = musicService.player.getCurrentPosition();//得到当前歌曲播放进度(秒)
- int mMax = musicService.player.getDuration();//最大秒数
- int sMax = seekBar.getMax();//seekBar最大值,算百分比
- seekBar.setProgress(position * sMax / mMax);
- txtInfo.setText(setPlayInfo(position / 1000, mMax / 1000));
- } catch (Exception e) {
- e.printStackTrace();
- }
- } else {
- seekBar.setProgress(0);
- txtInfo.setText("播放已经停止");
- }
- }
- }
- //设置当前播放的信息
- private String setPlayInfo(int position, int max) {
- String info = "正在播放: " + musicService.songName + "\t\t";
- //笨办法 写完才想起可以用%的,但不想改了
- int pMinutes = 0;
- while (position >= 60) {
- pMinutes++;
- position -= 60;
- }
- String now = (pMinutes < 10 ? "0" + pMinutes : pMinutes) + ":"
- + (position < 10 ? "0" + position : position);
- int mMinutes = 0;
- while (max >= 60) {
- mMinutes++;
- max -= 60;
- }
- String all = (mMinutes < 10 ? "0" + mMinutes : mMinutes) + ":"
- + (max < 10 ? "0" + max : max);
- return info + now + " / " + all;
- }
MusicThread的实现:
- class MusicThread implements Runnable {
- @Override
- public void run() {
- while (true)
- try {
- musicHandler.sendMessage(new Message());
- Thread.sleep(1000);// 每间隔1秒发送一次更新消息
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- }
Android音乐播放器的开发实例的更多相关文章
-
android音乐播放器开发教程
android音乐播放器开发教程 Android扫描sd卡和系统文件 Android 关于录音文件的编解码 实现米聊 微信一类的录音上传的功能 android操作sdcard中的多媒体文件——音乐列表 ...
-
Android音乐播放器源码(歌词.均衡器.收藏.qq5.0菜单.通知)
一款Android音乐播放器源码,基本功能都实现了 qq5.0菜单(歌词.均衡器.收藏.qq5.0菜单.通知) 只有向右滑动出现,菜单键和指定按钮都还没有添加. 源码下载:http://code.66 ...
-
一款非常简单的android音乐播放器源码分享给大家
一款非常简单的android音乐播放器源码分享给大家,该应用虽然很小,大家常用的播放器功能基本实现了,可能有点还不够完善,大家也可以自己完善一下,源码在源码天堂那里已经有了,大家可以到那里下载学习吧. ...
-
Android音乐播放器开发
今日看书,看到这个播放器,我就写了个例子,感觉还行,这个播放器能播放后缀是.MP3的音乐,这个例子在main.xml设置listView的时候,注意:android:id="@+id/and ...
-
android 音乐播放器
本章以音乐播放器为载体,介绍android开发中,通知模式Notification应用.主要涉及知识点Notification,seekbar,service. 1.功能需求 完善音乐播放器 有播放列 ...
-
Android音乐播放器的设计与实现
目录 应用开发技术及开发平台介绍 应用需求分析 应用功能设计及其描述 应用UI展示 一.应用开发技术及平台介绍 ①开发技术: 本系统是采用面向对象的软件开发方法,基于Android studio开发平 ...
-
[ 原创 ]学习笔记-做一个Android音乐播放器是遇到的一些困难
最近再做一个安卓的音乐播放器,是实验室里学长派的任务,我是在eclipse上进行开发的,由于没有android的基础,所以做起来困难重重. 首先是布局上的困难 1.layout里的控件属性不熟悉 2. ...
-
【竞品分析】Android音乐播放器的竞品分析
迄今为止最长的一篇博客,各位看官笑纳~~ 本次分析基于Android平台,选取了几款我体验过的播放器进行比较分析.主要分为两类,一类是大而全的,功能全面,可满足用户管理歌曲.导入导出歌单等多方面需求, ...
-
一个简单的Android音乐播放器
Android小白的期末作业 Android期末项目,仅用作学习使用,在线音乐部分只获取了网易云热歌榜,API来自鼻子亲了脸 传送门: GitHub 参考: anddiencn 实现功能 展示出本地的 ...
随机推荐
-
GitHub上值得关注的iOS开源项目
1.AFNetworking地址:https://github.com/AFNetworking/AFNetworking用于网络请求 2.JSONKit地址:https://github.com/j ...
-
SpringMVC学习笔记(四)
一.Controller接受网页参数. 1.使用方法的形参来接受 //使用基本类型和字符串来接受 @RequestMapping(value="/param2.do") publi ...
-
一种奇特的DEDE隐藏后门办法
转自:http://www.91ri.org/6462.html 一种奇特的DEDE隐藏后门办法 单位某站用的dedecms,今天被某黑阔getshell了,提交到了wooyun. 为了还原黑阔入 ...
-
hdu 2053 Switch Game 水题一枚,鉴定完毕
Switch Game Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
-
python 一遍式四则运算
#!/usr/bin/python #coding: utf-8 INTEGER = 'INTEGER' PLUS = '+' MINUS = '-' MUL = '*' DIV = '/' LC = ...
-
MySQL高可用架构之MHA 原理与实践
MHA简介 关于MHA MHA(Master HA)是一款开源的MySQL的高可用程序,它为MySQL主从复制架构提供了automating master failover 功能.MHA在监控到mas ...
-
Android开发学习之路--Activity之初体验
环境也搭建好了,android系统也基本了解了,那么接下来就可以开始学习android开发了,相信这么学下去肯定可以把android开发学习好的,再加上时而再温故下linux下的知识,看看androi ...
-
python实现简单的登录管理
import json,timeusername=[]userpasswd=[]def login_success_file(name): try: f=open(r"F:/login_su ...
-
Python之while循环
1.While循环基础 2.While循环进阶 3.其他
-
主从复制跳过错误(未采用GTID)
DB:5.6.15 主从库没有采用GTID 案例1:从库对象不存在mysql> show slave status \G;*************************** 1. row * ...