热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android实现简易的音乐播放器

这篇文章主要为大家详细介绍了Android实现简易的音乐播放器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android实现简易的音乐播放器,供大家参考,具体内容如下

功能介绍

本次实验实现的是使用Andriod Studio开发一个简易的音乐播放器,所包含的功能有音乐的播放、暂停、上一曲、下一曲、音乐播放的进度以及手动拖动来控制音乐的播放进度。

实现过程

导入项目所需的音乐文件、图标、背景等

1.创建一个raw文件夹,将音乐文件导入到这个文件夹中,方便我们在项目中使用

2.在drawable中导入所需的图片、图标

设计UI界面

1.设计5个button控件,分别对应上一曲,下一曲,暂停,播放,退出

2.设计3个TextView,分别对应歌曲的介绍信息、歌曲的进度(歌曲的总时间和歌曲当前播放的时间)、歌曲的名字

service服务的编写
创建一个MusicService对象继承Service

MusicService所需要的成员变量

MyReceiver serviceReceiver;
Thread processThread;
AssetManager am;//是附件管理器,用于根据文件名找到文件所在并打开文件
String[] musics = new String[]{"legendsneverdie.mp3", "promise.mp3",
        "beautiful.mp3"};//默认显示的歌曲信息
MediaPlayer mPlayer;
// 当前的状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
int status = 0x11;
// 记录当前正在播放的音乐
int current = 0;

实现循环播放

public void onCreate() {
        super.onCreate();
        am = getAssets();

        // 创建BroadcastReceiver
        serviceReceiver = new MyReceiver();
        // 创建IntentFilter
        IntentFilter filter = new IntentFilter();
        filter.addAction(MainActivity.CTL_ACTION);
        registerReceiver(serviceReceiver, filter);


        // 创建MediaPlayer
        mPlayer = new MediaPlayer();
        // 为MediaPlayer播放完成事件绑定监听器
        mPlayer.setOnCompletionListener(new OnCompletionListener()
        {
            @Override
            public void onCompletion(MediaPlayer mp) {
                Log.d("musicService", "播放完成");
                current++;
                if (current >= 3) {
                    current = 0;
                }
                // 准备并播放音乐
                prepareAndPlay(musics[current]);
                //发送广播通知Activity更改文本框
                Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
                sendIntent.putExtra("current", current);
                sendIntent.putExtra("currentTime", mPlayer.getCurrentPosition());
                sendIntent.putExtra("totalTime", mPlayer.getDuration());
                // 发送广播,将被Activity组件中的BroadcastReceiver接收到
                sendBroadcast(sendIntent);

            }
        });
    private void prepareAndPlay(String music) {
        try {
            // 打开指定音乐文件
            AssetFileDescriptor afd = am.openFd(music);
            mPlayer.reset();
            // 使用MediaPlayer加载指定的声音文件。
            mPlayer.setDataSource(afd.getFileDescriptor(),
                    afd.getStartOffset(), afd.getLength());
            // 准备声音
            mPlayer.prepare();
            // 播放
            mPlayer.start();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

实现刷新进度条

processThread = new Thread(new Runnable() {
        @Override
        public void run() {
            while (true) {
                if (status == 0x12) {
                    try {
                        Thread.sleep(1000);
                        Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
                        sendIntent.putExtra("current", current);
                        sendIntent.putExtra("currentTime", mPlayer.getCurrentPosition());
                        sendIntent.putExtra("totalTime", mPlayer.getDuration());
                        // 发送广播,将被Activity组件中的BroadcastReceiver接收到
                        sendBroadcast(sendIntent);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    });
    processThread.start();
}

广播通信接收器的实现(用于实现和activity的通信)

public class MyReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(final Context context, Intent intent) {
        int cOntrol= intent.getIntExtra("control", -1);
        Log.d("musicReceiver", "收到广播, cOntrol=" + control);
        switch (control) {
            // 播放或暂停
            case 1:
                // 原来处于没有播放状态
                if (status == 0x11) {
                    // 准备并播放音乐
                    prepareAndPlay(musics[current]);
                    status = 0x12;
                }
                // 原来处于播放状态
                else if (status == 0x12) {
                    // 暂停
                    mPlayer.pause();
                    // 改变为暂停状态
                    status = 0x13;
                }
                // 原来处于暂停状态
                else if (status == 0x13) {
                    // 播放
                    mPlayer.start();
                    // 改变状态
                    status = 0x12;
                }
                break;
                // 下一首
            case 2:
                if (status == 0x12 || status == 0x13) {
                    mPlayer.stop();
                    if (current + 1 >= musics.length) {
                        current = 0;
                    } else {
                        current++;
                    }
                    prepareAndPlay(musics[current]);
                    status = 0x12;
                    break;
                }
                // 上一首
            case 3:
                if (status == 0x12 || status == 0x13) {
                    mPlayer.stop();
                    if (current - 1 <0) {
                        current = musics.length - 1;
                    } else {
                        current--;
                    }
                    prepareAndPlay(musics[current]);
                    status = 0x12;
                }

        }

        // 广播通知Activity更改图标、文本框
        Intent sendIntent = new Intent(MainActivity.UPDATE_ACTION);
        sendIntent.putExtra("update", status);
        sendIntent.putExtra("current", current);
        // 发送广播,将被Activity组件中的BroadcastReceiver接收到
        sendBroadcast(sendIntent);
    }
}

activity的实现

初始化和动态绑定接收器

// 获取界面中显示歌曲标题、作者文本框
TextView title, author, currentTime, totalTime;
// 播放/暂停、停止按钮
ImageButton play;
ImageView lastMusic, nextMusic;
// 进度条
ProgressBar progressBar;

ActivityReceiver activityReceiver;

public static final String CTL_ACTION =
        "org.xr.action.CTL_ACTION";
public static final String UPDATE_ACTION =
        "org.xr.action.UPDATE_ACTION";
// 定义音乐的播放状态,0x11代表没有播放;0x12代表正在播放;0x13代表暂停
int status = 0x11;
String[] titleStrs = new String[]{"Legends Never Die", "约定", "美丽新世界"};
String[] authorStrs = new String[]{"英雄联盟", "周蕙", "伍佰"};

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    // 获取程序界面界面中的两个按钮
    play = (ImageButton) this.findViewById(R.id.play);
    lastMusic = this.findViewById(R.id.lastMusic);
    nextMusic = this.findViewById(R.id.nextMusic);
    title = (TextView) findViewById(R.id.title);
    author = (TextView) findViewById(R.id.author);
    currentTime = findViewById(R.id.currentTime);
    totalTime = findViewById(R.id.totalTime);
    progressBar = findViewById(R.id.progressBar);


    // 为两个按钮的单击事件添加监听器
    play.setOnClickListener(this);
    lastMusic.setOnClickListener(this);
    nextMusic.setOnClickListener(this);

    activityReceiver = new ActivityReceiver();
    // 创建IntentFilter
    IntentFilter filter = new IntentFilter();
    // 指定BroadcastReceiver监听的Action
    filter.addAction(UPDATE_ACTION);
    // 注册BroadcastReceiver
    registerReceiver(activityReceiver, filter);

    Intent intent = new Intent(this, MusicService.class);
    // 启动后台Service
    startService(intent);
}

设置activity的广播接收器(接收service发送过来的广播)

public void onReceive(Context context, Intent intent) {
    // 获取Intent中的update消息,update代表播放状态
    int update = intent.getIntExtra("update", -1);
    // 获取Intent中的current消息,current代表当前正在播放的歌曲
    int current = intent.getIntExtra("current", -1);
    int totalPosition = intent.getIntExtra("totalTime", -1);
    int currentPosition = intent.getIntExtra("currentTime", -1);
    Log.d("activityReceiver", "收到广播");
    Log.d("activityReceiver", "current:" + current + " totalPosition:" + totalPosition + " currentPosition:" + currentPosition + " update:" + update);
    if (current >= 0) {
        title.setText(titleStrs[current]);
        author.setText(authorStrs[current]);
    }

    if (totalPosition >= 0) {
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("mm:ss", Locale.CHINA);
        Date date = new Date(totalPosition);
        String formatTime = simpleDateFormat.format(date);
        totalTime.setText(formatTime);
    }

    if (currentPosition >= 0) {
        double process = ((double)currentPosition / totalPosition)*100;
        Log.d("activityReceiver", "当前进度:" + (double)currentPosition/totalPosition);
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("mm:ss", Locale.CHINA);
        Date date = new Date(currentPosition);
        String formatTime = simpleDateFormat.format(date);
        progressBar.setProgress((int) process);
        currentTime.setText(formatTime);
    }

    switch (update) {
        case 0x11:
            play.setImageResource(R.drawable.play);
            status = 0x11;
            break;
        // 控制系统进入播放状态
        case 0x12:
            // 播放状态下设置使用暂停图标
            play.setImageResource(R.drawable.pause);
            // 设置当前状态
            status = 0x12;
            break;
        // 控制系统进入暂停状态
        case 0x13:
            // 暂停状态下设置使用播放图标
            play.setImageResource(R.drawable.play);
            // 设置当前状态
            status = 0x13;
            break;
    }
}

实现图标的点击功能

// 创建Intent
    Intent intent = new Intent("org.xr.action.CTL_ACTION");
    switch (source.getId()) {
        // 按下播放/暂停按钮
        case R.id.play:
            intent.putExtra("control", 1);
            break;
        case R.id.lastMusic:
            intent.putExtra("control", 3);
        case R.id.nextMusic:
            intent.putExtra("control", 2);
    }
    // 发送广播,将被Service组件中的BroadcastReceiver接收到
    sendBroadcast(intent);
}

结果展示

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • Linux 系统启动故障排除指南:MBR 和 GRUB 问题
    本文详细介绍了 Linux 系统启动过程中常见的 MBR 扇区和 GRUB 引导程序故障及其解决方案,涵盖从备份、模拟故障到恢复的具体步骤。 ... [详细]
  • 本文介绍了如何使用jQuery根据元素的类型(如复选框)和标签名(如段落)来获取DOM对象。这有助于更高效地操作网页中的特定元素。 ... [详细]
  • 本文将详细介绍如何使用剪映应用中的镜像功能,帮助用户轻松实现视频的镜像效果。通过简单的步骤,您可以快速掌握这一实用技巧。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文介绍如何在 Xcode 中使用快捷键和菜单命令对多行代码进行缩进,包括右缩进和左缩进的具体操作方法。 ... [详细]
  • 本文介绍了一款用于自动化部署 Linux 服务的 Bash 脚本。该脚本不仅涵盖了基本的文件复制和目录创建,还处理了系统服务的配置和启动,确保在多种 Linux 发行版上都能顺利运行。 ... [详细]
  • 在Linux系统中配置并启动ActiveMQ
    本文详细介绍了如何在Linux环境中安装和配置ActiveMQ,包括端口开放及防火墙设置。通过本文,您可以掌握完整的ActiveMQ部署流程,确保其在网络环境中正常运行。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 如何在WPS Office for Mac中调整Word文档的文字排列方向
    本文将详细介绍如何使用最新版WPS Office for Mac调整Word文档中的文字排列方向。通过这些步骤,用户可以轻松更改文本的水平或垂直排列方式,以满足不同的排版需求。 ... [详细]
  • 本文总结了在使用Ionic 5进行Android平台APK打包时遇到的问题,特别是针对QRScanner插件的改造。通过详细分析和提供具体的解决方法,帮助开发者顺利打包并优化应用性能。 ... [详细]
  • 理解存储器的层次结构有助于程序员优化程序性能,通过合理安排数据在不同层级的存储位置,提升CPU的数据访问速度。本文详细探讨了静态随机访问存储器(SRAM)和动态随机访问存储器(DRAM)的工作原理及其应用场景,并介绍了存储器模块中的数据存取过程及局部性原理。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 几何画板展示电场线与等势面的交互关系
    几何画板是一款功能强大的物理教学软件,具备丰富的绘图和度量工具。它不仅能够模拟物理实验过程,还能通过定量分析揭示物理现象背后的规律,尤其适用于难以在实际实验中展示的内容。本文将介绍如何使用几何画板演示电场线与等势面之间的关系。 ... [详细]
author-avatar
wiggin
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有