基于Android的音乐播放器毕业论文.doc
毕业设计(论文)标 题: 基于Android的音乐播放器 学生姓名: 唐虹霞 系 部: 电子信息系 专 业: 软件技术 班 级: 软件1101班 指导教师: 湛桂枝 湖南汽车工程职业学院教务处制目录摘 要 当今社会的生活节奏越来越快+Android系统上一的款音乐播放器应用程序的设计与实现进行讨论。Android是一个开源的系统,它底层是基于Linux的操作系统,本论文的音乐播放器采用了Android开源系统技术,利用Java语言和Eclipse编辑工具对播放器进行编写。同时给出了详细的系统设计过程、部分界面图及主要功能运行流程图,本文还对高度过程中遇到的问题和解决方法进行了详细的讨论,该音乐播放器集播放、暂停、停止、上一首、下一首、音量调节、歌词显示等功能于一体,性能良好,在Android系统中能独立运行。该播放器还拥有对手机文件浏览器的访问功能、歌曲播放模式、以及歌词开闭状态的友好设置.MP3的全名是MPEG Audio Layer-3,是一种声音文件的压缩格式,由于本播放器只限于应用层程序的探讨,所以对具体的压缩算法不作深究。关键词:Android ; 开源系统 ; Eclipse ; 音乐播放器1系统需求分析1.1功能需求1.1.1播放器的基本控制需求根据项目的目标,我们可获得项目系统的基本需求,一下从不同的角度来描述系统的需求,并且使用用例图来描述,系统的功能需求,我们分成四部分来概括,即播放器的基本控制需要,播放列表管理需求,播放器友好性需求和播放器扩展卡需求。播放暂停停止上一曲下一曲 图1.1 播放器基本控制图 在播放器正在运行时,用户单击“播放”按钮,播放器将播放选中的播放列表中的音乐,并同时显示当前进度;当歌曲未暂停或停止时,用户单击“暂停”按钮,播放器将进入暂停状态;播放器正在播放或暂停时,用户单击“停止”按钮,播放器将停止播放或暂停时,用户点击“上一首”或者“下一首”按钮,播放器将播放上一首或下一首歌曲。1.1.2功能需求(时序图)分析从时序图中可以看出播放器中信息发送和调用过程主界面播放列表播放界面点击进入按钮选择歌曲播放进入播放列表播放后台serviceServic播放页面发送参数,调用后台service播放,返回播放界面图1.2 音乐播放器的时序图1.2 系统结构图和流程图主界面是否显示全部音乐播放列表是否点击要播放的歌曲音乐播放程序结束图1.3 音乐播放器系统流程图1.3系统界面需求播放器界面要求布局合理,简约,颜色舒适,控制按钮友好,能让用户一眼就能看出各个按钮的作用及操作方法。能实现要求的功能,但又不重复。要考虑到不同手机屏幕大小可能不一样,应尽量兼容大多数手机屏幕尺寸,使之显示无障碍。1.4系统性能需求根据Android手机系统要求无响应时间为5秒,所以就有如下性能要求:1、 当要求歌曲播放时,程序响应时间最长不能超过5秒2、 当要求歌曲暂停时,程序响应时间最长不能超过5秒3、 当要求歌曲停止时,程序响应时间最长不能超过5秒4、 当要求歌曲上/下一首时,程序响应时间最长不能超过5秒5、 界面切换响应时间最长不能超过3秒1.5运行环境需求操作系统:Windows xp支持环境:Android 2.1版本开发环境:Eclipse 3.5 ADT 0.952 Android项目介绍2.1什么是Android2.1.1Android的介绍Android一词的本义指“机器人”,同时也是Google于2007年11月5日宣布的基于Linux平台的开源手机操作系统的名称,该平台由操作系统、中间件、用户界面和应用软件组成,号称是首个为移动终端打造的真正开放和完整的移动软件。Android是基于Linux内核的软件平台和操作系统,早期由Google开发(在华注册商标名为“安致”),后由开放手机联盟(Open Handset Alliance)开发。它采用了软件堆层(software stack,又名以软件叠层)的架构,主要分为三部分。低层以Linux内核工作为基础,只提供基本功能;其他的应用软件则由各公司自行开发,以Java作为编写程序的一部分。另外,为了推广此技术,Google和其他几十个手机公司建立开发手机联盟。Android在未公开之前常被传闻为Google电话或gPhone。大多传闻认为Google开发的是自己的手机电话产品,而不是一套软件平台。到了2010年1月,Google开始发表自家品牌手机电话的Nexus One。2.1.2 Android Features 特性:Android 系统有如下的几大特性1. 应用程序框架 支持组建的重用与替换2. Dalvik 虚拟机 专门为移动设备做了优化3. 内部集成浏览器 该浏览器基于开源的WebKit引擎4. 优化的图形库 包括2D和3D图形库,3D图形库基于OpenGl ES5. SQLite 用作结构化的数据存储6. 多媒体支持 包括常见的音频、视频和静态印象文件格式(如MPEG4,H.264,MP3,AAC,AMR,JPG,PNG,GIF)7. GSM电话 (依赖于硬件)8. 蓝牙Bluetooth,EDGE,3G,安定WIFI(依赖于硬件)9. 照相机,GPS,指南针,和加速度计(依赖于硬件)10. 丰富的开发环境 包括设备模拟器,调试工具,内存及性能分析图表,和Eclipse集成开发环境插件2.1.3 Android 基本框架(Android Architecture)图2. 1Android Architecture(android 软件结构图)其软件层次结构自上而下分为以下几个层次第一、 应用程序(Application)第二、 应用程序框架(Application Framework)第三、 各种库(Libraries)和Android运行环境(RunTime)第四、 操作系统层(1) ApplicationApplication Android 会同一核心应用程序包一起发布,该应用程序包包括email客户端,SMS短消息程序,日历,地图,浏览器,联系人管理程序等。所有的应用程序都是用JAVA编写的。(2) Application FrameWork 开发者完全可以访问核心应用程序所使用的API框架。该应用程序架构用来简化组件软件的重用;任何一个应用程序都可以发布它的功能块并且任何其他的应用程序都可以使用其所发布的功能块(不过的遵循框架的安全性限制)。该应用程序重用机制使得组件可以被用户替换。所有的应用程序都由一系列的服务和系统组成,包括:a 可扩展的视图(Views)可以用来建应用程序,包括列表(lists),网格(Grids),文本框(Text boxes),按钮(button),甚至包括另一个可嵌入的web浏览器。b 内容管理器(Content Providers)使得应用程序可以访问另一个应用程序的数据(如联系人的数据库),或者共享它们自己的数据。c 资源管理器(Resource Manager)提供非代码资源的访问,如本地字符串,图形,和分层文件(Layout Files)。d 通知管理器(Notification Manager)使得应用程序生可以在状态栏中显示客户通知信息。e 活动类管理器(Activity Manager)用来管理应用程序生命周期并提供常用的导航回退功能。(3) Libraries库Android包括一个被Android系统中各种不同组件所使用的C/C+库集。该库通过Android应用程序框架为开发者提供服务。(4) Android RuntimeAndroid包括了一个核心库,该核心库提供了JAVA编程语言核心库的大多数功能。 每一个Android应用程序都在它自己的进程中运行,都拥有一个独立的Dalvik虚拟机实例。Dalvik是针对于同时高效地运行多个VMS来实现的。Dalvik虚拟机执行.dex的Dalvik可执行文件,该格式文件针对最小内存使用做了优化。该虚拟机是基于寄存器的,所有的类都经由JAVA汇编器编译,然后通过SDK中的dx工具转化成.dex格式由虚拟机执行。 Dalvik虚拟机依赖于linux的一些功能,比如线程机制和底层内存管理机制。(5) Linux内核Android的核心系统服务依赖于Linux2.6内核,如安全性,内存管理,进程管理,网络协议和驱动模型。Linux内核也同时作为硬件和软件堆栈之间的硬件抽象层。2.2 Android应用软件开发的核心技术2.2.1 Android 应用开发语言JAVA是Android上的主流开发语言,各种支持也最完善。Android发布的API和SDK也都主要是面向JAVA开发人员的。我的播放器实例也是用JAVA开发的。2.2.2 Android 应用程序组件Android 应用程序最大的一个特点就是每个应用都由若干组件构成。而且各个应用程序的组件还可以相互调用(需要在代码中设置相应权限)。例如假设我们需要编写一个内嵌于网页的程序,那么我们无需亲自编写浏览器,而是可以直接调用系统自带的Chrome Lite浏览器组件。又如假设我们编写了两个应用程序,其中一个程序中有列表显示图标的组件,那么我们在另一个应用程序中就可以使用该组件,而不需要在编写该功能。也就是说,不同于我们平常编写的程序,Android的应用程序不是单一入口,而是可以从代码的不同位置启动。这就要依靠组件来实现。Android应用程序的组件有以下四种:1. Activity一个Activity实际上表现出来的就是一个程序的界面。这个界面可能有许多不同的形式,比如有列表形式,有文字形式,有图像格子等形式。一个应用程序也可能有不止一个Activity。比如一耳光短消息的程序,需要有列表显示所有消息的Activity,还至少需要一个编辑短信的Activity,还可能需要设置、菜单等界面。每个Activity都有一个默认的窗口用于绘制界面,一般情况下,它是充满全屏的,但是我们可以通过程序控制它,使其小于屏幕大小,也可以控制其浮动于其他窗口之上(如Android中的Toast信息)。Activity中显示出来的内容其实是由view堆叠形成的,view是由View基类派生出来的各种形式的界面元素,如ListView,GridView等等,通过组合这些view,就形成了一个可见的Activity。我们可以用Activity.setContentView()方法来设置一个Activity中所要显示的view。2. ServiceService是一种没有界面的组件,同我们平常在Windows或Linux系统中所理解的“服务”一样,Android上的service也是运行在后台的,运行时间可以从系统启动到系统关闭为止。对于许多应用程序来说,service组件式至关重要的。以下载管理器为例,用户肯定希望能够在切换到其他程序时仍然在后台继续下载,这就需要用到service完成下载工作。而显示下载进度、已下载文件等信息则由Activity完成,同时我们还希望控制下载过程,如停止、继续等,这就需要Activity与service之间进行通信。Android允许activity连接到或者绑定到service上,这样就可以让用户在界面上控制后台service的行为。3. Broadcast receiverBroadcast意为广播消息,Android系统自身会发出许多广播消息,如电量不足,用户修改了全局设置等,各个应用程序也以向整个系统广播消息,如文件下载。Broadcast receiver则是程序中专门用来接收广播消息的组件,它没有界面显示,只负责监听、接收广播消息,然后根据程序逻辑,启动其他组件去处理广播消息。它可以启动一个Activity来同用户交互,也可以使用NotificationManager向用户发出通知。NitificationManager的 通知形式有许多种,不如说振动、响铃、闪LED灯,还可以在Android界面的消息栏处显示信息。4. Content providersContent provider是一种为其它应用程序提供数据的组件。这些数据可以保存在文件中,也可以保存在SQLite数据库中,还可以以其他任何有效的方式保存。ContentProvider类为我们提供了一系列保存数据的方法供我们调用,可以很方便地存储数据。其它应用程序在需要使用这些数据时,可以用ContentResolver对象,使用其中的方法。ContentResolver可以与任何contentprovider交互,获取其中的数据。只需要某个组件,Android就会尽力保证它的可用;如果该组件还没有启动,就启动它,如果该组件所属的应用程序还没拥有一个进程,则为它启动一个进程,如果系统资源不足以容纳一个新的进程,则将后台的某个进程杀死为新进程腾出资源。2.3 Android开发环境的搭配任何事物要运行,都要有它的环境,Android也有它的环境才能够运行,下面介绍Android的开发环境配置.搭建开发环境需要的软件:操作系统:Windows XP或Linux软件包:Android SDK(Software Development kit Java Development kit) 、ADT(Android Develoopment Tool)IDE环境:Eclipse IDE+ADT Eclipse3.3以上JDK:Java Runtime Environment虚拟机 、(JDK)Java Development kit安装步骤如下:第一步:安装Java虚拟机sun-java6-jdk版本第二步:安装Eclipse3.5工具 官网http:/www.eclipse.org/downloads/选择版本(图4.2):进行安装图 2.2 第三步:安装Android SDK:首先要下载SDK,可以在这个网址下载Http:/ ,选择相应的系统进行安装图 2.3 第四步:安装Android ADT插件运行Eclipse,选择help->install new software 选择add,将会弹出一个框图 2.4 点击OK,选择要安装的软件,占next,然后选择接受协议,直到安装成功,重启Eclipse软件,设置Android SDK Home,Window->refrence图 2.5 在SDK location中输入SDK Tools路径:D:android softwareandroid-sdk-windows 点击OK这样Android环境就已经搭建成功了。2.4 Android模拟器中sd卡的创建和文件的上传在Android模拟器中,初始时没有sdcard的,为了存放音乐文件,需要自己创建sdcard,我们先来看看如何创建sdcard以及文件上传。2.4.1 SD卡的创建通过Android的Tools来创建sd卡并导入进来1. 打开dos,windows+R键打开。转换到你的sdk安装包的tools目录下,如我的目录如下:E:android-sdk-windowstools 2. 创建sd卡。在命令行中输入:mksdcard 1024M E:androidsdcardsdcard.img(这个1024M就是我们要创建的SD卡的大小,还是根据自己的需要来输入,最后面的那个E:androidsdcardsdcard.img就是所设置的SD存放路径,这里也可以根据我们自己的需要改变路径的,但要注意路径中不能有空格出现)3. 在Eclipse里面创建模拟器,并以该SD开来创建4. 创建完成之后,启动该模拟器。可以在AndroidSDK and AVD Manager 里面指定模拟器并启动。5. 同样打开Eclipse的DDMS-àFileExplorer 查看sd卡是否安装成功。2.4.2向sd卡里面上传文件及异常解决右边两个小手机按钮就是传输文件的按钮了,用它来实现文件的传入和传出了,向左的表示把手机里的文件导出,向右的表示把文件导入到手机里面去。这里和容易出现异常。异常如下:异常一:Failed to push selection:Invalid argument出现此异常,多半是文件所在的目录里面有中文或者文件的名字是中文导致的。建议修改一下试试。还有就是观察自己的操作权限是不是不够,如果是,drwxrx 那就是可以的。如果,sd卡是在eclipse里面系统自动创建的,试着重启一下IDE。异常二:Failed to push the item(s)出现这个异常是因为传输超时造成的。可以修改。 打开windowsPreferenceandroidDDMS 把ADB connection tine out 项的超时时间改大点。3音乐播放器的详细设计3.1 播放器主界面的设计和说明Android的每一个可视化界面,都有其唯一的布局配置文件,该文件里面有各种布局方式,各种资源文件如图像,文字,颜色的引用,程序在运行时,可以通过代码对配置文件进行读取。这样就可以形成不同的可视化界面和绚丽的效果。主界面布局文件内容如下:<?xml version="1.0" encoding="utf-8"?> <AbsoluteLayout xmlns:android=" android:id="+id/AbsoluteLayout01" android:layout_width="wrap_content" android:layout_height="fill_parent" android:background="drawable/backgrund" android:sharedUserId="com.android.activitygrouptest" > 此处的ListView作用为列出歌曲<ListView android:id="id/android:list" android:layout_width="fill_parent" android:layout_height="fill_parent" android:layout_weight="1" android:drawSelectorOnTop="false"/><LinearLayout android:layout_width="fill_parent" android:layout_height="wrap_content" android:gravity="center" android:orientation="horizontal" >此处的ImageButton用于第一曲 <ImageButton android:id="+id/music_lasted" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginRight="5dp" android:src="drawable/latest1" />此处的ImageButton用于上一曲 <ImageButton android:id="+id/music_rewind" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginRight="5dp" android:src="drawable/rewind" />此处的ImageButton用于播放与暂停 <ImageButton android:id="+id/music_play" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginRight="5dp" android:src="drawable/play1" />此处的ImageButton用于下一曲 <ImageButton android:id="+id/music_foward" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginRight="5dp" android:src="drawable/foward" />此处的ImageButton用于最后一曲 <ImageButton android:id="+id/music_next" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="drawable/next1" /> </LinearLayout>主界面为:3.2 Android媒体播放原理Android源程序已经为我们封装了一个接口类,叫MediaPlayer。该接口在执行前需要一个数据源,通过如下四个步骤: 第一步:MediaPlayer.reset(); 复位操作,为以后的媒体准备播放做准备。 第二步:传入歌曲数据源,这里也是一个数据源:MediaPlayer.setDataSource(String path);参数需要一个路径,该接口类在收到路径后将其转化为数据源。 第三步:准备播放数据源,调用MediaPlayer.prepare(); 第四步:MediaPlayer.start();播放数据源,该数据源播放后,不再于程序有关,需要调用MediaPlayer.stop()方法让其停止或MediaPlayer.start();让其启动。3.3 播放功能实现细节为了实现音乐播放器的基本功能:音乐播放、暂停、停止、上一首、下一首。这里我用事件监听,详细如下:3.3.1 播放器播放、暂停、停止等功能 图 3.3.1 如图5.4所示:各按钮水平放置,用的是布局中的线性布局LinearLayout,用方法setOrientation(horizontal);将其设置为水平。LinearLayout中依次放置第一曲、上一曲、播放与暂停、下一曲、最后一曲按钮属性。配置文件代码结构如下:<LinearLayout><ImageButton ></ImageButton><ImageButton ></ImageButton><ImageButton ></ImageButton><ImageButton ></ImageButton><ImageButton ></ImageButton></LinearLayout> android的媒体播放原理:android源程序已经为我们封装好了一个接口类,叫MediaPlayer。该接口在执行行需要一个数据源,通过如下四个步骤:第一步:MediaPlayer.reset();复位操作,为以后的媒体准备播放作准备。 第二步:传入歌曲数据源,这里也是一个数据流 MediaPlayer.setDataSource(String path);参数需要传一个路径,该接口类在收到路径后将其转化为数据源。第三步:准备播放数据源,调用方法MediaPlayer.Prepare();第四步:MediaPlayer.Start();播放数据源,该数据源播放后,不再与程序 有关,需要用MediaPlayer.Stop()方法让其停止或MediaPlayer.Start();让其启动;歌曲数据源的获得:要获得数据源,需要我们提供给接口类一个歌曲存放路径,而这个路径已经被事先放到数据库里了,我们只需要进行数据库进行操作。将里面的路径取出。这里将调用PlayRackActivity的一个方法query();代码算法如下:public String query() ContentResolver cr = getContentResolver();Uri uri = DBProvider.CONTENT_URI;String projection = "path" ;String selection = "fileName=?"String selectionArgs = selectName ;Cursor c=cr.query(uri,projection,selection, selectionArgs, null);if (c.moveToFirst() String path = c.getString(0);return path;return null; 方法getContentResolver是包android.content.ContextWrapper里面的方法,它会自动的调用ContentProvider中的关联数据库,我们需用这个方法就可以操作数据库。数据库操作需要一个Uri 。Uri类似一个网络连接的Url。Uri是数据库唯一的标识,不同的Uri可以操作不同的数据库表。在查询数据库时需要传入一个当前播放的歌曲名字,通过歌曲名字来查询歌曲的路径。4 Android数据库设计 数据结构组织和数据库文件设计要根据不同用途,使用要求等,来决定数据的整体组织形式等一系列问题。数据库,顾名思义,是存放数据的仓库。只不过这个仓库是在计算机存储设备上,而且数据是按一定的格式存放的。数据库中的数据按一定数据模型组织、描述和存储,具有较小的重复度、较高的数据独立性和易扩展性,并且可以被在一定范围内的各种用户共享。在涉及数据库的软件开发中,需要根据有待解决的问题性质、规模,以及所采用的前端程序创建工具等,做出合适的数据库类型选择4.1 数据库及字段属性设计4.1.1 字段设计(表、图)file_table 主要是保存歌曲名字、类型、路径字段说明:Id 歌曲id号 fileName 歌曲名字 filePath 歌曲路径 sort 歌曲类型(表4.1、图4.1.1) Android自带一个MediaStore封闭类 专门来存储媒体信息通过Uri EXTERNAL_CONTENT_URI 来访问SDcard中的歌曲详细信息。存放媒体信息如TITLE(标题)、ARTIST(艺术家)、ALBUM(专辑)、SIZE(大小 )(表4.2、图4.1.2) 属性数据类型允许空主键_IdINTEGERNot nullPrimary keyfileNameTEXTNot nullfilePathTEXTsortINTEGER 表4.1 属性数据类型允许空主键_IDINTEGERNot nullPrimary keyTITLETEXTNot nullARTISTTEXTALBUMTEXTSIZELONG表4.2 图4.1.1 歌曲列表 图4.1.2 歌曲详细 4.1.2 音乐播放器E-R图音乐播放器 E-R(实体-联系:entity-relation图)图(图4.1.3) 图4.1.3 E-R图 4.2 数据库连接 这里将介绍如何进行Android数据库连接,Android中自带SQLite数据库,这是一个十分小型的数据库,这样正适合Android这种移动平台使用。Android数据库存储的位置在data/data/<项目文件夹>/databases/目录下 Android是利用ContentProvider作为内容提供商,SQLiteOpenHelper数据库帮助类来进行对数据库的创建和操作。通过Context.getContentResolver()方法直接对数据库进行操作。程序中数据库类为DBHelper extends SQLiteOpenHelper(继承关系),内容提供类DBProvider extends ContentProvider(继承关系)4.2.1 创建数据库 Android 提供了标准的数据库创建方式。继承SQLiteOpenHelper ,实现onCreate 和 onUpgrade 两个方法,有个好处就是便于数据库版本的升级,连接数据库的算法如下:public DBHelper(Context context) / 创建数据库super(context, DATABASE_NAME,null, DATABASE_VERSION);public void onCreate(SQLiteDatabase db) / 创建时调用 db.execSQL(DATABASE_CREATE);public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) / 版本更新时调用db.execSQL("DROP TABLE IF EXISTS File_Table");onCreate(db); Override / 在ContextProvider运行时自动创建数据库 public boolean onCreate() dbOpenHelper= new DBHelper(getContext(); return true;数据库如果创建不成功则抛出FIleNotFoundException异常4.2.2 操作数据库 Android对数据库的操作主要有插入、删除、更新、查询操作,在进行任何操作时都必须指定一个Uri,才能对相应的表进行数据操作。/数据库删除操作Overridepublic int delete(Uri arg0, String arg1, String arg2) SQLiteDatabase db = dbOpenHelper.getWritableDatabase();try Db.delete(FileColumn.TABLE, arg1, arg2); catch (Exception ex) ex.printStackTrace();return 1;Override/数据库插入操作public Uri insert(Uri uri, ContentValues values) SQLiteDatabase db = dbOpenHelper.getWritableDatabase();long count = 0;try count = db.insert(FileColumn.TABLE, null, values); catch (Exception ex) ex.printStackTrace(); if (count > 0) return uri;else return null;/ 数据库更新操作Overridepublic int update(Uri uri, ContentValues values, String selection,String selectionArgs) SQLiteDatabase db = dbOpenHelper.getWritableDatabase();int i = 0;try i = db.update(FileColumn.TABLE, valu