本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下
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 的内容如下:
我们注意到有2个Activity,1个Service,还有读写外部存储的权限声明
3、CoverActivity.java的代码如下:这是个全屏的启动画面,2秒后会跳转到PlayerActivity
01 | package app.android.elfplayer; |
03 | import android.app.Activity; |
04 | import android.content.Intent; |
05 | import android.os.Bundle; |
06 | import android.os.Handler; |
07 | import android.view.Window; |
08 | import android.view.WindowManager; |
10 | public class CoverActivity extends Activity { |
11 | /** Called when the activity is first created. */ |
13 | public void onCreate(Bundle savedInstanceState) { |
14 | super.onCreate(savedInstanceState); |
15 | getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); |
16 | requestWindowFeature(Window.FEATURE_NO_TITLE); |
17 | setContentView(R.layout.cover); |
19 | new Handler().postDelayed(new Runnable(){ |
23 | Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class); |
24 | CoverActivity.this.startActivity(mainIntent); |
25 | CoverActivity.this.finish(); |
4、PlayerActivity.java的代码如下:
001 | package app.android.elfplayer; |
003 | import android.app.Activity; |
004 | import android.content.ComponentName; |
005 | import android.content.Context; |
006 | import android.content.Intent; |
007 | import android.content.ServiceConnection; |
008 | import android.os.Bundle; |
009 | import android.os.Handler; |
010 | import android.os.IBinder; |
011 | import android.os.Message; |
012 | import android.os.RemoteException; |
013 | import android.util.Log; |
014 | import android.view.View; |
015 | import android.widget.ImageButton; |
016 | import android.widget.SeekBar; |
017 | import android.widget.SeekBar.OnSeekBarChangeListener; |
019 | public class PlayerActivity extends Activity { |
021 | public static final int PLAY = 1; |
022 | public static final int PAUSE = 2; |
024 | ImageButton imageButtonFavorite; |
025 | ImageButton imageButtonNext; |
026 | ImageButton imageButtonPlay; |
027 | ImageButton imageButtonPre; |
028 | ImageButton imageButtonRepeat; |
029 | SeekBar musicSeekBar; |
031 | IServicePlayer iPlayer; |
032 | boolean isPlaying = false; |
033 | boolean isLoop = false; |
036 | public void onCreate(Bundle savedInstanceState) { |
037 | super.onCreate(savedInstanceState); |
038 | setContentView(R.layout.player); |
040 | imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite); |
041 | imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext); |
042 | imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay); |
043 | imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre); |
044 | imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat); |
045 | musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar); |
047 | bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE); |
048 | startService(new Intent(PlayerActivity.this, MusicService.class)); |
050 | imageButtonPlay.setOnClickListener(new View.OnClickListener() { |
053 | public void onClick(View v) { |
054 | Log.i("yao", "imageButtonPlay -> onClick"); |
059 | } catch (RemoteException e) { |
062 | imageButtonPlay.setBackgroundResource(R.drawable.pause_button); |
068 | } catch (RemoteException e) { |
071 | imageButtonPlay.setBackgroundResource(R.drawable.play_button); |
077 | musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() { |
080 | public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { |
084 | public void onStartTrackingTouch(SeekBar seekBar) { |
088 | public void onStopTrackingTouch(SeekBar seekBar) { |
089 | if (iPlayer != null) { |
091 | iPlayer.seekTo(seekBar.getProgress()); |
092 | } catch (RemoteException e) { |
099 | handler.post(updateThread); |
102 | private ServiceConnection conn = new ServiceConnection() { |
103 | public void onServiceConnected(ComponentName className, IBinder service) { |
104 | Log.i("yao", "ServiceConnection -> onServiceConnected"); |
105 | iPlayer = IServicePlayer.Stub.asInterface(service); |
108 | public void onServiceDisconnected(ComponentName className) { |
112 | Handler handler = new Handler() { |
114 | public void handleMessage(Message msg) { |
118 | private Runnable updateThread = new Runnable() { |
121 | if (iPlayer != null) { |
123 | musicSeekBar.setMax(iPlayer.getDuration()); |
124 | musicSeekBar.setProgress(iPlayer.getCurrentPosition()); |
125 | } catch (RemoteException e) { |
129 | handler.post(updateThread); |
5、其中用到的IServicePlayer.aidl,放在和Java文件相同的包中,内容如下:
01 | package app.android.elfplayer; |
02 | interface IServicePlayer{ |
07 | int getCurrentPosition(); |
08 | void seekTo(int current); |
09 | boolean setLoop(boolean loop); |
一旦你写好了这个IServicePlayer.aidl文件,ADT会自动帮你在gen目录下生成IServicePlayer.java文件
6、MusicService.java的内容如下:
01 | package app.android.elfplayer; |
03 | import android.app.Service; |
04 | import android.content.Intent; |
05 | import android.media.MediaPlayer; |
06 | import android.os.IBinder; |
07 | import android.os.RemoteException; |
08 | import android.util.Log; |
10 | public class MusicService extends Service { |
14 | public static MediaPlayer mPlayer; |
16 | public boolean isPause = false; |
18 | IServicePlayer.Stub stub = new IServicePlayer.Stub() { |
21 | public void play() throws RemoteException { |
26 | public void pause() throws RemoteException { |
31 | public void stop() throws RemoteException { |
36 | public int getDuration() throws RemoteException { |
37 | return mPlayer.getDuration(); |
41 | public int getCurrentPosition() throws RemoteException { |
42 | return mPlayer.getCurrentPosition(); |
46 | public void seekTo(int current) throws RemoteException { |
47 | mPlayer.seekTo(current); |
51 | public boolean setLoop(boolean loop) throws RemoteException { |
58 | public void onCreate() { |
59 | Log.i(tag, "MusicService onCreate()"); |
60 | mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3")); |
64 | public IBinder onBind(Intent intent) { |
7、其它代码和资源可以参见本讲附带的源代码,编译并运行程序,查看结果:
最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。
最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素
阅读(1399) | 评论(0) | 转发(0) |