本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下AIDL和Remote Service的价值和使用方法,你把这个例子跑一边,体会一下就OK了。下面的例子是我正在准备的项目实例中的一部分。
首先说明一下我们面临的问题,如果看不懂下面的描述请看前面的课程:
第一、我们知道在AndroId中如果需要进行音乐播放,最方面的方法就是使用自带的MediaPlayer对象,如果我们在Activity中控制MediaPlayer对象进行播放,那么一旦你打开了另外一个程序譬如浏览器,那么歌声就会立刻停止,这当然不是我们需要的结果。 我们需要的是在做其他事情的同时能够在后台听歌,于是我们就需要把对MediaPlayer对象的操作放在后台Service中去。
第二、我们已经把对MediaPlayer的操作转移到Service中去了,按照我们以前的做法,我们在Activity中发送一个Intent对象给Service对象,在Intent中传送播放啊、暂停啊一类的信息给Service,这样Service就知道该怎么做了。这一切看起来很美好,可是现在出了一个新问题,那就是我想在Activity中显示一个进度条,这个进度条要跟着Service中的MediaPlayer中的歌曲进度同步向前走,而且如果我点击进度条中的某一个位置,还想让歌曲跳转到新的时间点继续播放,这个,该怎么实现?
第三、我们需要在Activity中操作Service中的MediaPlayer对象,就好像这个对象是自己的一样。我们可以采用Android接口定义语言 AIDL(Android Interface Definition Language)技术:
1、把Service中针对MediaPlayer的操作封装成一个接口(.aidl文件)
2、在Service中建个子类实现这接口的存根(stub)对象
3、并在onBind()方法中返回这个存根对象。
4、在Activity中使用绑定服务的方式连接Service,但是不用Intent来传递信息,而是在ServiceConnection的onServiceConnected方法里,获得Service中Stub对象的客户端使用代理。我们通过操作Activity中的代理就可以达到操作Service中的MediaPlayer对象的目的。这样我们就可以想用本地对象一样操作Service中的对象了,那么进度条一类的需求自然也就迎刃而解。
下面的例子,并不是专门为本讲准备的,所以有些无关代码,而且没加注释,请见谅(本例完整讲解会放在项目实训中,正在准备):
1、新建一个项目 App_elfPlayer ,启动Activity是个启动画面:CoverActivity
2、AndroidManifest.xml 的内容如下:
  1. <?xml version="1.0" encoding="utf-8"?>

  2. <manifest package="app.android.elfplayer" xmlns:android="http://schemas.android.com/apk/res/android" android:versioncode="1" android:versionname="1.0">

  3.        <uses -sdk="" android:minsdkversion="7">

  4.        <uses -permission="" android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses>


  5.        <application android:label="@string/app_name" android:icon="@drawable/icon">

  6.                <activity android:name=".CoverActivity">

  7.                        <intent -filter="">

  8.                                <action android:name="android.intent.action.MAIN">

  9.                                <category android:name="android.intent.category.LAUNCHER">

  10.                        </category></action></intent>

  11.                </activity>

  12.                <activity android:name=".PlayerActivity">

  13.                </activity>

  14.                <service android:name=".MusicService" android:enabled="true">

  15.                </service>

  16.        </application>


  17. </uses></manifest>


我们注意到有2个Activity,1个Service,还有读写外部存储的权限声明3、CoverActivity.java的代码如下:这是个全屏的启动画面,2秒后会跳转到PlayerActivity
  1. package app.android.elfplayer;


  2. import android.app.Activity;

  3. import android.content.Intent;

  4. import android.os.Bundle;

  5. import android.os.Handler;

  6. import android.view.Window;

  7. import android.view.WindowManager;


  8. public class CoverActivity extends Activity {

  9.        /** Called when the activity is first created. */

  10.        @Override

  11.        public void onCreate(Bundle savedInstanceState) {

  12.                super.onCreate(savedInstanceState);

  13.                getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);

  14.                requestWindowFeature(Window.FEATURE_NO_TITLE);

  15.                setContentView(R.layout.cover);


  16.                new Handler().postDelayed(new Runnable(){


  17.                 @Override

  18.                 public void run() {

  19.                     Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);

  20.                     CoverActivity.this.startActivity(mainIntent);

  21.                     CoverActivity.this.finish();

  22.                 }


  23.                }, 2000);


  24.        }

  25. }


4、PlayerActivity.java的代码如下:
  1. package app.android.elfplayer;


  2. import android.app.Activity;

  3. import android.content.ComponentName;

  4. import android.content.Context;

  5. import android.content.Intent;

  6. import android.content.ServiceConnection;

  7. import android.os.Bundle;

  8. import android.os.Handler;

  9. import android.os.IBinder;

  10. import android.os.Message;

  11. import android.os.RemoteException;

  12. import android.util.Log;

  13. import android.view.View;

  14. import android.widget.ImageButton;

  15. import android.widget.SeekBar;

  16. import android.widget.SeekBar.OnSeekBarChangeListener;


  17. public class PlayerActivity extends Activity {


  18.        public static final int PLAY = 1;

  19.        public static final int PAUSE = 2;


  20.        ImageButton imageButtonFavorite;

  21.        ImageButton imageButtonNext;

  22.        ImageButton imageButtonPlay;

  23.        ImageButton imageButtonPre;

  24.        ImageButton imageButtonRepeat;

  25.        SeekBar musicSeekBar;


  26.        IServicePlayer iPlayer;

  27.        boolean isPlaying = false;

  28.        boolean isLoop = false;        


  29.        @Override

  30.        public void onCreate(Bundle savedInstanceState) {

  31.                super.onCreate(savedInstanceState);

  32.                setContentView(R.layout.player);


  33.                imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);

  34.                imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);

  35.                imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);

  36.                imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);

  37.                imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);

  38.                musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);


  39.                bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);

  40.                startService(new Intent(PlayerActivity.this, MusicService.class));


  41.                imageButtonPlay.setOnClickListener(new View.OnClickListener() {


  42.                        @Override

  43.                        public void onClick(View v) {

  44.                                Log.i("yao", "imageButtonPlay -> onClick");


  45.                                if (!isPlaying) {

  46.                                        try {

  47.                                                iPlayer.play();

  48.                                        } catch (RemoteException e) {

  49.                                                e.printStackTrace();

  50.                                        }

  51.                                        imageButtonPlay.setBackgroundResource(R.drawable.pause_button);

  52.                                        isPlaying = true;


  53.                                } else {

  54.                                        try {

  55.                                                iPlayer.pause();

  56.                                        } catch (RemoteException e) {

  57.                                                e.printStackTrace();

  58.                                        }

  59.                                        imageButtonPlay.setBackgroundResource(R.drawable.play_button);

  60.                                        isPlaying = false;

  61.                                }

  62.                        }

  63.