Service 的两种启动方式,以及在音乐播放中的使用

 首先,简要介绍一下service

   service就是android系统中的服务,它有这么几个特点:它无法与用户直接进行交互、它必须由用户或者其他程序显式的启动、它的优先级比较高, 它比处于前台的应用优先级低,但是比后台的其他应用优先级高,这就决定了当系统因为缺少内存而销毁某些没被利用的资源时,它被销毁的概率很小。

 

bindService是绑定Service服务,执行service服务中的逻辑流程。

service通过Context.startService()方法开始,通过Context.stopService()方法停止;也可以通过Service.stopSelf()方法或者Service.stopSelfResult()方法来停止自己。只要调用一次stopService()方法便可以停止服务,无论之前它被调用了多少次的启动服务方法。

客户端建立一个与Service的连接,并使用此连接与Service进行通话,通过Context.bindService()方法来绑定服务,Context.unbindService()方法来关闭服务。多个客户端可以绑定同一个服务,如果Service还未被启动,bindService()方法可以启动服务。

    startService()和bindService()两种模式是完全独立的。你可以绑定一个已经通过startService()方法启动的服务。例如:一 个后台播放音乐服务可以通过startService(intend)对象来播放音乐。可能用户在播放过程中要执行一些操作比如获取歌曲的一些信息,此时 activity可以通过调用bindServices()方法与Service建立连接。这种情况下,stopServices()方法实际上不会停止 服务,直到最后一次绑定关闭。

    这2种模式不是完全分离的。你可以可以绑定到一个通过startService()启动的服务。如 一个intent想要播放音乐,通过startService() 方法启动后台播放音乐的service。然后,也许用户想要操作播放器或者获取当前正在播放的乐曲的信息,一个activity就会通过 bindService()建立一个到此service的连接. 这种情况下 stopService() 在全部的连接关闭后才会真正停止service。

   

bindService启动流程

context.bindService()  ——> onCreate()  ——> onBind()  ——> Service running  ——> onUnbind()  ——> onDestroy()  ——> Service stop

onBind()将返回给 客户端一个IBind接口实例,IBind允许客户端回调服务的方法,比如得到Service的实例、运行状态或其他操作。这个时候把调用者 (Context,例如Activity)会和Service绑定在一起,Context退出了,Srevice就会调用 onUnbind->onDestroy相应退出。

所以调用bindService的生命周期为:onCreate --> onBind(只一次,不可多次绑定) --> onUnbind --> onDestory。

在Service每一次的开启关闭过程中,只有onStart可被多次调用(通过多次startService调用),其他onCreate,onBind,onUnbind,onDestory在一个生命周期中只能被调用一次。

 

整个生命周期

service 的整个生命周期是在onCreate()和onDestroy()方法之间。和activity一样,在onCreate()方法里初始化,在 onDestroy()方法里释放资源。例如,一个背景音乐播放服务可以在onCreate()方法里播放,在onDestroy()方法里停止。

活动的生命周期

service 的活动生命周期是在onStart()之后,这个方法会处理通过startServices()方法传递来的Intent对象。音乐service可以通 过开打intent对象来找到要播放的音乐,然后开始后台播放。注: service停止时没有相应的回调方法,即没有onStop()方法,只有onDestroy()销毁方法。

任何服务无论它怎样建立,默认客户端都可以连接,所以任何service都能够接收onBind()和onUnbind()方法

onCreate()方法和onDestroy()方法是针对所有的services,无论它们是否启动,通过Context.startService()和Context.bindService()方法都可以访问执行。然而,只有通过startService()方法启动service服务时才会调用onStart()方法。

 

如果一个service允许别人绑定,那么需要实现以下额外的方法:

      IBinder onBind(Intent intent)

      boolean onUnbind(Intent intent)

      void onRebind(Intent intent)

onBind()回调方法会继续传递通过bindService()传递来的intent对象

onUnbind()会处理传递给unbindService()的intent对象。如果service允许绑定,onBind()会返回客户端与服务互相联系的通信句柄(实例)。

如果建立了一个新的客户端与服务的连接,onUnbind()方法可以请求调用onRebind()方法。

 

经过上面的简介,那么,当我们需要实现一个音乐服务,在状态栏包含通知,同时在很多界面都可以调用音乐服务器的进度,音乐名等信息的功能时,我们已经有思路了,那就是用startservice 开启一个音乐服务,然后在需要获取信息的界面,采用bindservice 获取音乐信息,需要注意的是,我们在此处,启动音乐服务不能采用bindservice ,因为此模式启动,当首次绑定服务的界面销毁时,服务也跟随销毁了。

 

如下

/**
 * 佛祖保佑        永无BUG
 * 佛曰:
 * 程序园里程序天,程序天里程序员;
 * 程序猿人写程序,又拿程序换肉钱。
 * 肉饱继续桌前坐,饱暖还是桌前眠;
 * 半迷半醒日复日,码上码下年复年。
 * 但愿叱咤互联世,不愿搬砖码当前;
 * 诸葛周瑜算世事,我算需求得加钱。
 * 别人笑我忒直男,我说自己是程猿;
 * 但见成都府国内,处处地地程序员。
 * Created by 水东流 on 2018/8/13.
 */

public class MusicService extends Service {
    List<String> musicPath = new ArrayList<>();//播放地址
    //标记当前歌曲的序号
    private int index = 0;

    public MusicService() {
        iniMediaPlayerFile(0);
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }

    /*设置锁屏界面*/
    @Override
    public void onCreate() {
        super.onCreate();
      /*  receiver = new BroadcastReceiver() {
            @Override
            public void onReceive(Context context, Intent intent) {
                if (intent.getAction() == Intent.ACTION_SCREEN_OFF) {
                    Log.e(TAG, "收到锁屏广播");
                    Intent lockscreen = new Intent(MusicService.this, MusicSuoPingActivity.class);
                    lockscreen.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    startActivity(lockscreen);
                }
            }
        };
        IntentFilter filter = new IntentFilter();
        filter.addAction(Intent.ACTION_SCREEN_OFF);
        registerReceiver(receiver, filter);*/
    }

    public class MyBinder extends Binder {
        /**
         * 是否正在播放
         */
        public boolean isPlay() {
            if (mMediaPlayer.isPlaying()) {
                return true;
            }
            return false;
        }

        /**
         * 更换播放地址
         *
         * @param url
         */
        public void changeUrl(List<String> url) {
            musicPath = url;
            if (mMediaPlayer != null) {
                mMediaPlayer.reset();
            }
            index = 0;
            iniMediaPlayerFile(index);
        }

        /**
         * 播放音乐
         */
        public void playMusic() {
            if (!mMediaPlayer.isPlaying()) {
                //如果还没开始播放,就开始
                mMediaPlayer.start();
            }
        }

        /**
         * 暂停播放
         */
        public void pauseMusic() {
            if (mMediaPlayer.isPlaying()) {
                //如果还没开始播放,就开始
                mMediaPlayer.pause();
            }
        }

        /**
         * reset
         */
        public void resetMusic() {
            if (!mMediaPlayer.isPlaying()) {
                //如果还没开始播放,就开始
                mMediaPlayer.reset();
                iniMediaPlayerFile(index);
            }
        }

        /**
         * 关闭播放器
         */
        public void closeMedia() {
            if (mMediaPlayer != null) {
                mMediaPlayer.stop();
                mMediaPlayer.release();
            }
        }

        /**
         * 下一首
         */
        public void nextMusic() {
            if (mMediaPlayer != null && index < musicPath.size() && index >= 0) {
                //切换歌曲reset()很重要很重要很重要,没有会报IllegalStateException
                mMediaPlayer.reset();
                //这里的if只要是为了不让歌曲的序号越界
                if (index == musicPath.size() - 1) {
                    index = 0;
                } else {
                    index = index + 1;
                }
                iniMediaPlayerFile(index);
                playMusic();
            }
        }

        /**
         * 上一首
         */
        public void preciousMusic() {
            if (mMediaPlayer != null && index < musicPath.size() && index > 0) {
                mMediaPlayer.reset();
                if (index == 1) {
                    index = musicPath.size() - 1;
                } else {
                    index = index - 1;
                }
                iniMediaPlayerFile(index);
                playMusic();
            }
        }

        /**
         * 获取歌曲长度
         **/
        public int getProgress() {
            return mMediaPlayer.getDuration();
        }

        /**
         * 获取播放位置
         */
        public int getPlayPosition() {
            return mMediaPlayer.getCurrentPosition();
        }

        /**
         * 播放指定位置
         */
        public void seekToPositon(int msec) {
            mMediaPlayer.seekTo(msec);
        }


    }

    /**
     * 添加file文件到MediaPlayer对象并且准备播放音频
     */
    private void iniMediaPlayerFile(int position) {
        if (musicPath == null || musicPath.size() == 0) return;
        //获取文件路径
        try {
            //此处的两个方法需要捕获IO异常
            //设置音频文件到MediaPlayer对象中
            mMediaPlayer.setDataSource(musicPath.get(position));
            //让MediaPlayer对象准备
            mMediaPlayer.prepare();
        } catch (IOException e) {
            Log.d(TAG, "设置资源,准备阶段出错");
            e.printStackTrace();
        }
    }

    private BroadcastReceiver receiver;
    //初始化MediaPlayer
    public MediaPlayer mMediaPlayer = new MediaPlayer();
    private static final String TAG = "MediaService";
    private MyBinder mBinder = new MyBinder();

}

 

启动界面代码

/**启动音乐服务*/
        /** 单击按钮时启动服务 */
        Intent intent = new Intent(mcontext,
                MusicService.class);
        startService(intent);

获取音乐服务信息界面的代码

/**
 * 绑定
 */
private void initMusic() {
    mServiceConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            mMyBinder = (MusicService.MyBinder) service;
            viewHolder.seekbar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
                @Override
                public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
                    //这里很重要,如果不判断是否来自用户操作进度条,会不断执行下面语句块里面的逻辑,然后就会卡顿卡顿
                    if (fromUser) {
                        mMyBinder.seekToPositon(seekBar.getProgress());
                    }
                }

                @Override
                public void onStartTrackingTouch(SeekBar seekBar) {

                }

                @Override
                public void onStopTrackingTouch(SeekBar seekBar) {

                }
            });
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
        }
    };
    MediaServiceIntent = new Intent(mcontext, MusicService.class);
    bindService(MediaServiceIntent, mServiceConnection, BIND_AUTO_CREATE);
}

绑定之后就可以通过 mMyBinder来获取和操作服务中信息数据了

case R.id.precious:
    mMyBinder.preciousMusic();
    break;
case R.id.play:
    if (mMyBinder.isPlay()) {
        mMyBinder.pauseMusic();
        viewHolder.play.setImageResource(R.mipmap.zant1);
    } else {
        mMyBinder.playMusic();
        viewHolder.play.setImageResource(R.mipmap.zant2);
    }
    break;
case R.id.next:
    mMyBinder.nextMusic();
    break;

 

 

 

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值