Android--音乐播放器

简介: 版权声明:本文为博主原创文章,转载请标明出处。 https://blog.csdn.net/chaoyu168/article/details/51150983 1、什么是Open Core?Open Core 是 Android 多媒体框架的核心,所有 Android 平台的音频、视频的采用以及播放等操作,都是通过它来实现。
版权声明:本文为博主原创文章,转载请标明出处。 https://blog.csdn.net/chaoyu168/article/details/51150983

1、什么是Open Core?

Open Core 是 Android 多媒体框架的核心,所有 Android 平台的音频、视频的采用以及播放等操作,都是通过它来实现。

 

2、Open Core 的具体功能

  • 1、多媒体文件的播放、下载,包括:3GPP,MPEG-4,AAC 和 MP3 containers。
  • 2、流媒体文件的下载、实时播放,包括:3GPP,HTTP 和 RTSP/RTP。
  • 3、动态视频和静态图像的编码、解码,例如:MPEG-4,H.263 和 AVC (H.264),JPEG。
  • 4、语音编码格式:MP3,AAC,AAC+。
  • 5、视频和图像格式:3GPP,MPEG-4 和 JPEG。
  • 6、视频会议:基于 H324-M 标准。

Open Core 是一个多媒体的框架,从宏观上来看,它主要包含了两大方面的内容:

  • 1、PVPPlayer:提供媒体播放器的功能,完成各种音频(Audio)、视频(Video)流的回放(Playback)功能。
  • 2、PVAuthor:提供媒体流记录的功能,完成各种音频、视频流以及静态图像的捕获功能。

3、Mediaplayer 介绍 

MediaPlayer 类可以用来播放音频、视频和流媒体,MediaPlayer 包含了 Audio 和 Video 的播放功能,在 Android 的界面上, Music 和 Video 两个应用程序都是调用 MediaPlayer 实现的。下面两张图是 MediaPlayer  在 Android 和 Open Core 的示例图:

 

下面是图例显示 MediaPlayer 的生命周期

3.1、MediaPlayer 常用方法介绍

4、完整音乐播放例子演示

实现的功能:

1.暂停/播放、下一首/上一首,点击某一首时播放

2.支持拖动进度条快进

3.列表排序

4.来电话时,停止播放,挂断后继续播放

5.可在后台播放


界面:

main.xml:

<pre name="code" class="html"><span style="font-size:14px;"><?xml version="1.0" encoding="utf-8"?>  
<LinearLayout xmlns:Android="http://schemas.android.com/apk/res/android"  
    Android:orientation="vertical"  
    Android:layout_width="fill_parent"  
    Android:layout_height="fill_parent"  
    >  
    <TextView    
        Android:id="@+id/name"  
        Android:layout_width="fill_parent"   
        Android:layout_height="wrap_content"   
        />  
    <SeekBar  
        Android:id="@+id/seekBar"  
        Android:layout_width="fill_parent"   
        Android:layout_height="wrap_content"   
        Android:layout_marginBottom="5dp"  
        />  
    <LinearLayout  
        Android:layout_width="fill_parent"   
        Android:layout_height="wrap_content"   
        Android:layout_marginBottom="20dp"  
        >  
        <Button  
            Android:layout_width="40dp"   
            Android:layout_height="40dp"   
            Android:text="|◀"  
            Android:onClick="previous"  
            />  
        <Button  
            Android:id="@+id/pp"  
            Android:layout_width="40dp"   
            Android:layout_height="40dp"   
            Android:text="▶"  
            Android:onClick="pp"  
            />  
        <Button  
            Android:layout_width="40dp"   
            Android:layout_height="40dp"   
            Android:text="▶|"  
            Android:onClick="next"  
            />  
    </LinearLayout>  
    <ListView  
        Android:id="@+id/list"  
        Android:layout_width="fill_parent"   
        Android:layout_height="fill_parent"   
        />  
</LinearLayout></span>  


 
 

item.xml:

<span style="font-size:14px;"><?xml version="1.0" encoding="utf-8"?>  
<LinearLayout  
    xmlns:Android="http://schemas.android.com/apk/res/android"  
    Android:layout_width="fill_parent"  
    Android:layout_height="wrap_content"  
    Android:padding="10dp"  
    >  
    <TextView  
        Android:id="@+id/mName"  
        Android:layout_width="fill_parent"  
        Android:layout_height="wrap_content"  
        Android:textSize="15sp"  
        />  
</LinearLayout></span><span style="font-size: 18px;"> </span>

MainActivity:
<pre name="code" class="java"><span style="font-size:14px;">import java.io.File;  
import java.util.ArrayList;  
import java.util.Collections;  
import java.util.HashMap;  
import java.util.List;  
import java.util.Map;  
  
import Android.app.Activity;  
import Android.content.BroadcastReceiver;  
import Android.content.Context;  
import Android.content.Intent;  
import Android.content.IntentFilter;  
import Android.media.MediaPlayer;  
import Android.media.MediaPlayer.OnCompletionListener;  
import Android.os.Bundle;  
import Android.os.Environment;  
import Android.os.Handler;  
import Android.telephony.PhoneStateListener;  
import Android.telephony.TelephonyManager;  
import Android.view.View;  
import Android.widget.AdapterView;  
import Android.widget.Button;  
import Android.widget.ListView;  
import Android.widget.SeekBar;  
import Android.widget.SimpleAdapter;  
import Android.widget.TextView;  
import Android.widget.AdapterView.OnItemClickListener;  
import Android.widget.SeekBar.OnSeekBarChangeListener;  
  
public class MainActivity extends Activity {  
    private TextView nameTextView;  
    private SeekBar seekBar;  
    private ListView listView;  
    private List<Map<String, String>> data;  
    private int current;  
    private MediaPlayer player;  
    private Handler handler = new Handler();  
    private Button ppButton;  
    private boolean isPause;  
    private boolean isStartTrackingTouch;  
  
    public void onCreate(Bundle savedInstanceState) {   
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.main);  
  
        nameTextView = (TextView) findViewById(R.id.name);  
        seekBar = (SeekBar) findViewById(R.id.seekBar);  
        listView = (ListView) findViewById(R.id.list);  
        ppButton = (Button) findViewById(R.id.pp);  
          
        //创建一个音乐播放器   
        player = new MediaPlayer();  
  
        //显示音乐播放列表   
        generateListView();  
  
        //进度条监听器   
        seekBar.setOnSeekBarChangeListener(new MySeekBarListener());  
          
        //播放器监听器   
        player.setOnCompletionListener(new MyPlayerListener());  
  
        //意图过滤器   
        IntentFilter filter = new IntentFilter();  
          
        //播出电话暂停音乐播放   
        filter.addAction("Android.intent.action.NEW_OUTGOING_CALL");  
        registerReceiver(new PhoneListener(), filter);  
  
        //创建一个电话服务   
        TelephonyManager manager = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);  
          
        //监听电话状态,接电话时停止播放   
        manager.listen(new MyPhoneStateListener(), PhoneStateListener.LISTEN_CALL_STATE);  
    }  
  
      
    private final class MyPhoneStateListener extends PhoneStateListener {  
        public void onCallStateChanged(int state, String incomingNumber) {  
            pause();  
        }  
    }  
  
      
    private final class MyPlayerListener implements OnCompletionListener {  
        //歌曲播放完后自动播放下一首歌曲   
        public void onCompletion(MediaPlayer mp) {  
            next();  
        }  
    }  
  
      
    public void next(View view) {  
        next();  
    }  
  
      
    public void previous(View view) {  
        previous();  
    }  
  
      
    private void previous() {  
        current = current - 1 < 0 ? data.size() - 1 : current - 1;  
        play();  
    }  
  
      
    private void next() {  
        current = (current + 1) % data.size();  
        play();  
    }  
  
      
    private final class MySeekBarListener implements OnSeekBarChangeListener {  
        //移动触发   
        public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {  
        }  
  
        //起始触发   
        public void onStartTrackingTouch(SeekBar seekBar) {  
            isStartTrackingTouch = true;  
        }  
  
        //结束触发   
        public void onStopTrackingTouch(SeekBar seekBar) {  
            player.seekTo(seekBar.getProgress());  
            isStartTrackingTouch = false;  
        }  
    }  
  
      
    private void generateListView() {  
        List<File> list = new ArrayList<File>();  
          
        //获取sdcard中的所有歌曲   
        findAll(Environment.getExternalStorageDirectory(), list);  
          
        //播放列表进行排序,字符顺序   
        Collections.sort(list);  
  
        data = new ArrayList<Map<String, String>>();  
        for (File file : list) {  
            Map<String, String> map = new HashMap<String, String>();  
            map.put("name", file.getName());  
            map.put("path", file.getAbsolutePath());  
            data.add(map);  
        }  
  
        SimpleAdapter adapter = new SimpleAdapter(this, data, R.layout.item, new String[] { "name" }, new int[] { R.id.mName });  
        listView.setAdapter(adapter);  
  
        listView.setOnItemClickListener(new MyItemListener());  
    }  
  
    private final class MyItemListener implements OnItemClickListener {  
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {  
            current = position;  
            play();  
        }  
    }  
  
      
    private void play() {  
        try {  
            //重播   
            player.reset();  
              
            //获取歌曲路径   
            player.setDataSource(data.get(current).get("path"));  
              
            //缓冲   
            player.prepare();  
              
            //开始播放   
            player.start();  
              
            //显示歌名   
            nameTextView.setText(data.get(current).get("name"));  
              
            //设置进度条长度   
            seekBar.setMax(player.getDuration());  
  
            //播放按钮样式   
            ppButton.setText("||");  
  
            //发送一个Runnable, handler收到之后就会执行run()方法   
            handler.post(new Runnable() {  
                public void run() {  
                    // 更新进度条状态   
                    if (!isStartTrackingTouch)  
                        seekBar.setProgress(player.getCurrentPosition());  
                    // 1秒之后再次发送   
                    handler.postDelayed(this, 1000);  
                }  
            });  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
  
      
    private void findAll(File file, List<File> list) {  
        File[] subFiles = file.listFiles();  
        if (subFiles != null)  
            for (File subFile : subFiles) {  
                if (subFile.isFile() && subFile.getName().endsWith(".mp3"))  
                    list.add(subFile);  
                else if (subFile.isDirectory())//如果是目录   
                    findAll(subFile, list); //递归   
            }  
    }  
  
      
    public void pp(View view) {  
          
        //默认从第一首歌曲开始播放   
        if (!player.isPlaying() && !isPause) {  
            play();  
            return;  
        }  
  
        Button button = (Button) view;  
        //暂停/播放按钮   
        if ("||".equals(button.getText())) {  
            pause();  
            button.setText("▶");  
        } else {  
            resume();  
            button.setText("||");  
        }  
    }  
  
      
    private void resume() {  
        if (isPause) {  
            player.start();  
            isPause = false;  
        }  
    }  
  
      
    private void pause() {  
        if (player != null && player.isPlaying()) {  
            player.pause();  
            isPause = true;  
        }  
    }  
  
      
    private final class PhoneListener extends BroadcastReceiver {  
        public void onReceive(Context context, Intent intent) {  
            pause();  
        }  
    }  
  
      
    protected void onResume() {  
        super.onResume();  
        resume();  
    }  
}
</span>


 
 
注册权限:
<pre name="code" class="html"><?xml version="1.0" encoding="utf-8"?>  
<manifest xmlns:Android="http://schemas.android.com/apk/res/android"  
      package="cn.itcast.audio"  
      Android:versionCode="1"  
      Android:versionName="1.0">  
    <application Android:icon="@drawable/icon" android:label="@string/app_name">  
        <activity Android:name=".MainActivity"  
                  Android:label="@string/app_name">  
            <intent-filter>  
                <action Android:name="android.intent.action.MAIN" />  
                <category Android:name="android.intent.category.LAUNCHER" />  
            </intent-filter>  
        </activity>  
  
    </application>  
    <uses-sdk Android:minSdkVersion="8" />  
      
    <!-- 监听电话呼出 -->  
    <uses-permission Android:name="android.permission.PROCESS_OUTGOING_CALLS"/>  
    <!-- 监听电话状态改变 -->  
    <uses-permission Android:name="android.permission.READ_PHONE_STATE" />   
  
</manifest>  


 
 

 
 

目录
相关文章
|
缓存 网络协议 开发工具
庖丁解牛之-Android平台RTSP|RTMP播放器设计
我们在做Android平台RTSP或者RTMP播放器开发的时候,需要注意的点非常多,以下,以大牛直播SDK(官方)的接口为例,大概介绍下相关接口设计:
142 0
|
6月前
|
XML Java Android开发
Android Studio App开发之服务Service的讲解及实战(包括启动和停止,绑定与解绑,推送服务到前台实现音乐播放器,附源码)
Android Studio App开发之服务Service的讲解及实战(包括启动和停止,绑定与解绑,推送服务到前台实现音乐播放器,附源码)
797 0
|
前端开发 开发工具 Android开发
Android播放器之SurfaceView与GLSurfaceView
Surface的官方介绍:Handle onto a raw buffer that is being managed by the screen compositor,Surface是一个raw buffer的句柄,通过它在raw buffer上进行绘制,可以通过Surface获得一个Canvas。
267 0
|
8天前
|
Java 程序员 开发工具
Android|修复阿里云播放器下载不回调的问题
虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。
19 2
|
5月前
|
XML 存储 数据库
如何使用Android Studio创建一个基本的音乐播放器应用
如何使用Android Studio创建一个基本的音乐播放器应用
211 0
|
3月前
|
编解码 开发工具 Android开发
Android平台RTSP|RTMP播放器如何实现TextureView渲染
本文介绍了在Android平台上使用TextureView进行RTSP和RTMP视频流渲染的技术背景和实现方法。TextureView相较于SurfaceView具备更高性能、更强功能性和更灵活的绘制方式等优势,但也有必须在硬件加速环境下运行和较高内存占用等局限。文中详细展示了如何在SmartPlayerV2工程中创建和配置TextureView,并通过代码示例解释了如何根据视频分辨率信息调整显示比例,以及处理TextureView的各种生命周期回调。此外,还列举了该播放器SDK支持的多项高级功能,如多实例播放、多种编码格式支持、硬解码能力等,旨在帮助开发者更好地理解和实现高性能的直播播放器。
|
3月前
|
算法 数据处理 开发工具
Android平台RTSP|RTMP播放器如何回调YUV或RGB数据
在开发Android平台上的RTSP或RTMP播放器时,开发者不仅追求低延迟播放,还希望获取解码后的视频数据(如YUV或RGB格式),以便进行视觉算法分析。使用大牛直播SDK中的SmartPlayer,可在确保播放流畅的同时,通过设置外部渲染器(`SmartPlayerSetExternalRender`)来高效地回调原始视频数据。例如,对于RGBA数据,需实现`NTExternalRender`接口,并重写相关方法以处理数据和尺寸变化。同样地,对于I420(YUV)数据,也需要相应地实现接口以满足需求。这种方式使得开发者能在不影响常规播放功能的情况下,进行定制化的视频处理任务。
|
3月前
|
编解码 网络协议 开发工具
Android平台RTSP|RTMP直播播放器技术接入说明
大牛直播SDK自2015年发布RTSP、RTMP直播播放模块,迭代从未停止,SmartPlayer功能强大、性能强劲、高稳定、超低延迟、超低资源占用。无需赘述,全自研内核,行业内一致认可的跨平台RTSP、RTMP直播播放器。本文以Android平台为例,介绍下如何集成RTSP、RTMP播放模块。
144 0
|
5月前
|
API Android开发 UED
56. 【Android教程】媒体播放器:MediaPlayer
56. 【Android教程】媒体播放器:MediaPlayer
96 0
|
开发工具 Android开发 iOS开发
Android、iOS平台RTMP/RTSP播放器实现实时音量调节
介绍移动端RTMP、RTSP播放器实时音量调节之前,我们之前也写过,为什么windows播放端加这样的接口,windows端播放器在多窗口大屏显示的场景下尤其需要,尽管我们老早就有了实时静音接口,相对实时静音来说,播放端实时音量调节粒度更细,从[0, 100],用户体验更好。
200 1