Chinaunix首页 | 论坛 | 博客
  • 博客访问: 292920
  • 博文数量: 68
  • 博客积分: 1474
  • 博客等级: 上尉
  • 技术积分: 616
  • 用 户 组: 普通用户
  • 注册时间: 2011-02-12 12:07
文章分类

全部博文(68)

文章存档

2011年(68)

分类: 嵌入式

2011-05-04 20:51:35


本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下 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 的内容如下:

01
02
03    
04    
05 
06    
07        
08            
09                
10                
11            
12        
13        
14        
15        
16        
17    
18 
19

我们注意到有2个Activity,1个Service,还有读写外部存储的权限声明

3、CoverActivity.java的代码如下:这是个全屏的启动画面,2秒后会跳转到PlayerActivity

01package app.android.elfplayer;
02 
03import android.app.Activity;
04import android.content.Intent;
05import android.os.Bundle;
06import android.os.Handler;
07import android.view.Window;
08import android.view.WindowManager;
09 
10public class CoverActivity extends Activity {
11    /** Called when the activity is first created. */
12    @Override
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);
18 
19        new Handler().postDelayed(new Runnable(){
20 
21             @Override
22             public void run() {
23                 Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);
24                 CoverActivity.this.startActivity(mainIntent);
25                 CoverActivity.this.finish();
26             }
27 
28            }, 2000);
29 
30    }
31}

4、PlayerActivity.java的代码如下:

001package app.android.elfplayer;
002 
003import android.app.Activity;
004import android.content.ComponentName;
005import android.content.Context;
006import android.content.Intent;
007import android.content.ServiceConnection;
008import android.os.Bundle;
009import android.os.Handler;
010import android.os.IBinder;
011import android.os.Message;
012import android.os.RemoteException;
013import android.util.Log;
014import android.view.View;
015import android.widget.ImageButton;
016import android.widget.SeekBar;
017import android.widget.SeekBar.OnSeekBarChangeListener;
018 
019public class PlayerActivity extends Activity {
020 
021    public static final int PLAY = 1;
022    public static final int PAUSE = 2;
023 
024    ImageButton imageButtonFavorite;
025    ImageButton imageButtonNext;
026    ImageButton imageButtonPlay;
027    ImageButton imageButtonPre;
028    ImageButton imageButtonRepeat;
029    SeekBar musicSeekBar;
030 
031    IServicePlayer iPlayer;
032    boolean isPlaying = false;
033    boolean isLoop = false;
034 
035    @Override
036    public void onCreate(Bundle savedInstanceState) {
037        super.onCreate(savedInstanceState);
038        setContentView(R.layout.player);
039 
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);
046 
047        bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);
048        startService(new Intent(PlayerActivity.this, MusicService.class));
049 
050        imageButtonPlay.setOnClickListener(new View.OnClickListener() {
051 
052            @Override
053            public void onClick(View v) {
054                Log.i("yao", "imageButtonPlay -> onClick");
055 
056                if (!isPlaying) {
057                    try {
058                        iPlayer.play();
059                    } catch (RemoteException e) {
060                        e.printStackTrace();
061                    }
062                    imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
063                    isPlaying = true;
064 
065                } else {
066                    try {
067                        iPlayer.pause();
068                    } catch (RemoteException e) {
069                        e.printStackTrace();
070                    }
071                    imageButtonPlay.setBackgroundResource(R.drawable.play_button);
072                    isPlaying = false;
073                }
074            }
075        });
076 
077        musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {
078 
079            @Override
080            public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
081            }
082 
083            @Override
084            public void onStartTrackingTouch(SeekBar seekBar) {
085            }
086 
087            @Override
088            public void onStopTrackingTouch(SeekBar seekBar) {
089                if (iPlayer != null) {
090                    try {
091                        iPlayer.seekTo(seekBar.getProgress());
092                    } catch (RemoteException e) {
093                        e.printStackTrace();
094                    }
095                }
096            }
097        });
098 
099        handler.post(updateThread);
100    }
101 
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);
106        }
107 
108        public void onServiceDisconnected(ComponentName className) {
109        };
110    };
111 
112    Handler handler = new Handler() {
113        @Override
114        public void handleMessage(Message msg) {
115        };
116    };
117 
118    private Runnable updateThread = new Runnable() {
119        @Override
120        public void run() {
121            if (iPlayer != null) {
122                try {
123                    musicSeekBar.setMax(iPlayer.getDuration());
124                    musicSeekBar.setProgress(iPlayer.getCurrentPosition());
125                } catch (RemoteException e) {
126                    e.printStackTrace();
127                }
128            }
129            handler.post(updateThread);
130        }
131    };
132 
133}

5、其中用到的IServicePlayer.aidl,放在和Java文件相同的包中,内容如下:

01package app.android.elfplayer;
02interface IServicePlayer{
03    void play();
04    void pause();
05    void stop();
06    int getDuration();
07    int getCurrentPosition();
08    void seekTo(int current);
09    boolean setLoop(boolean loop);
10}

一旦你写好了这个IServicePlayer.aidl文件,ADT会自动帮你在gen目录下生成IServicePlayer.java文件

6、MusicService.java的内容如下:

01package app.android.elfplayer;
02 
03import android.app.Service;
04import android.content.Intent;
05import android.media.MediaPlayer;
06import android.os.IBinder;
07import android.os.RemoteException;
08import android.util.Log;
09 
10public class MusicService extends Service {
11 
12    String tag = "yao";
13 
14    public static MediaPlayer mPlayer;
15 
16    public boolean isPause = false;
17 
18    IServicePlayer.Stub stub = new IServicePlayer.Stub() {
19 
20        @Override
21        public void play() throws RemoteException {
22            mPlayer.start();
23        }
24 
25        @Override
26        public void pause() throws RemoteException {
27            mPlayer.pause();
28        }
29 
30        @Override
31        public void stop() throws RemoteException {
32            mPlayer.stop();
33        }
34 
35        @Override
36        public int getDuration() throws RemoteException {
37            return mPlayer.getDuration();
38        }
39 
40        @Override
41        public int getCurrentPosition() throws RemoteException {
42            return mPlayer.getCurrentPosition();
43        }
44 
45        @Override
46        public void seekTo(int current) throws RemoteException {
47            mPlayer.seekTo(current);
48        }
49 
50        @Override
51        public boolean setLoop(boolean loop) throws RemoteException {
52            return false;
53        }
54 
55    };
56 
57    @Override
58    public void onCreate() {
59        Log.i(tag, "MusicService onCreate()");
60        mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));
61    }
62 
63    @Override
64    public IBinder onBind(Intent intent) {
65        return stub;
66    }
67 
68}

7、其它代码和资源可以参见本讲附带的源代码,编译并运行程序,查看结果:

image

image

image

最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。

最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素

阅读(1399) | 评论(0) | 转发(0) |
0

上一篇:Android JNI实例

下一篇:Android消息机制(一)

给主人留下些什么吧!~~