Android AIDL和远程Service调用示例代码

Android:AIDL和远程Service调用

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

<?xml version="1.0" encoding="utf-8"?> <manifest package="app.android.elfplayer" xmlns:android="http://schemas.android.com/apk/res/android" android:versioncode="1" android:versionname="1.0"> <uses -sdk="" android:minsdkversion="7"> <uses -permission="" android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses> <application android:label="@string/app_name" android:icon="@drawable/icon"> <activity android:name=".CoverActivity"> <intent -filter=""> <action android:name="android.intent.action.MAIN"> <category android:name="android.intent.category.LAUNCHER"> </category></action></intent> </activity> <activity android:name=".PlayerActivity"> </activity> <service android:name=".MusicService" android:enabled="true"> </service> </application> </uses></manifest>

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

package app.android.elfplayer; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.os.Handler; import android.view.Window; import android.view.WindowManager; public class CoverActivity extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.cover); new Handler().postDelayed(new Runnable(){ @Override public void run() { Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class); CoverActivity.this.startActivity(mainIntent); CoverActivity.this.finish(); } }, 2000); } }

4、PlayerActivity.java的代码如下:

package app.android.elfplayer; import android.app.Activity; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.ServiceConnection; import android.os.Bundle; import android.os.Handler; import android.os.IBinder; import android.os.Message; import android.os.RemoteException; import android.util.Log; import android.view.View; import android.widget.ImageButton; import android.widget.SeekBar; import android.widget.SeekBar.OnSeekBarChangeListener; public class PlayerActivity extends Activity { public static final int PLAY = 1; public static final int PAUSE = 2; ImageButton imageButtonFavorite; ImageButton imageButtonNext; ImageButton imageButtonPlay; ImageButton imageButtonPre; ImageButton imageButtonRepeat; SeekBar musicSeekBar; IServicePlayer iPlayer; boolean isPlaying = false; boolean isLoop = false; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.player); imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite); imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext); imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay); imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre); imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat); musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar); bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE); startService(new Intent(PlayerActivity.this, MusicService.class)); imageButtonPlay.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Log.i("yao", "imageButtonPlay -> onClick"); if (!isPlaying) { try { iPlayer.play(); } catch (RemoteException e) { e.printStackTrace(); } imageButtonPlay.setBackgroundResource(R.drawable.pause_button); isPlaying = true; } else { try { iPlayer.pause(); } catch (RemoteException e) { e.printStackTrace(); } imageButtonPlay.setBackgroundResource(R.drawable.play_button); isPlaying = false; } } }); musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() { @Override public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { } @Override public void onStartTrackingTouch(SeekBar seekBar) { } @Override public void onStopTrackingTouch(SeekBar seekBar) { if (iPlayer != null) { try { iPlayer.seekTo(seekBar.getProgress()); } catch (RemoteException e) { e.printStackTrace(); } } } }); handler.post(updateThread); } private ServiceConnection conn = new ServiceConnection() { public void onServiceConnected(ComponentName className, IBinder service) { Log.i("yao", "ServiceConnection -> onServiceConnected"); iPlayer = IServicePlayer.Stub.asInterface(service); } public void onServiceDisconnected(ComponentName className) { }; }; Handler handler = new Handler() { @Override public void handleMessage(Message msg) { }; }; private Runnable updateThread = new Runnable() { @Override public void run() { if (iPlayer != null) { try { musicSeekBar.setMax(iPlayer.getDuration()); musicSeekBar.setProgress(iPlayer.getCurrentPosition()); } catch (RemoteException e) { e.printStackTrace(); } } handler.post(updateThread); } }; }

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

package app.android.elfplayer; interface IServicePlayer{ void play(); void pause(); void stop(); int getDuration(); int getCurrentPosition(); void seekTo(int current); boolean setLoop(boolean loop); }

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

6、MusicService.java的内容如下:

package app.android.elfplayer; import android.app.Service; import android.content.Intent; import android.media.MediaPlayer; import android.os.IBinder; import android.os.RemoteException; import android.util.Log; public class MusicService extends Service { String tag = "yao"; public static MediaPlayer mPlayer; public boolean isPause = false; IServicePlayer.Stub stub = new IServicePlayer.Stub() { @Override public void play() throws RemoteException { mPlayer.start(); } @Override public void pause() throws RemoteException { mPlayer.pause(); } @Override public void stop() throws RemoteException { mPlayer.stop(); } @Override public int getDuration() throws RemoteException { return mPlayer.getDuration(); } @Override public int getCurrentPosition() throws RemoteException { return mPlayer.getCurrentPosition(); } @Override public void seekTo(int current) throws RemoteException { mPlayer.seekTo(current); } @Override public boolean setLoop(boolean loop) throws RemoteException { return false; } }; @Override public void onCreate() { Log.i(tag, "MusicService onCreate()"); mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3")); } @Override public IBinder onBind(Intent intent) { return stub; } }

7、实现效果图:

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

以上就是Android AIDL和远程Service 的介绍和简单应用,后续继续补充相关知识,谢谢大家的支持!

时间: 2024-11-17 10:24:52

Android AIDL和远程Service调用示例代码的相关文章

Android AIDL和远程Service调用示例代码_Android

Android:AIDL和远程Service调用 本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命.所以我们干脆拿一个音乐播放器中进度条的实例来说明一下AIDL和Remote Service的价值和使用方法,你把这个例子跑一边,体会一下就OK了.下面的例子是我 正在准备的项目实例中的一部分. 首先说明一下我们面临的问题,如果看不懂下面的描述请看前面的课程: 第一.我们知道在AndroId中如果需要进行音乐播放,最方面的方法就是使用自带的MediaPlayer对象,如

Android TextView实现词组高亮的示例代码

本文介绍了Android TextView实现词组高亮的示例代码,分享给大家,具体如下: HighlightTextView Android文本高亮控件,基于View实现. 特点 文本高亮 单词自动换行 高亮词组保持在同一行显示 效果如下: 主要逻辑: 两个 Paint 负责绘制不同的文字 在每次绘制之前计算将要绘制的文本是否会超出屏幕宽度,如果超出则换行 protected void onDraw(Canvas canvas) { super.onDraw(canvas); float x_d

Android学习之远程绑定调用service

http://blog.csdn.net/q1234456gggg_jkjg/article/details/8479070 远程绑定调用service主要是用来不同进程的信息共享.就比如服务器和客户端,在服务器端设置好一个service提供方法或信息,然后客户端可以直接调用服务器端service提供方法或信息.这里有个前提是客户端必须有和服务器端一份一样的AIDL,然后服务器端在客户端使用的系统上有注册过(也就是安装运行过一次),之后客户端就可以远程绑定调用服务器端的service了. 具体的

Android Dialog 对话框详解及示例代码_Android

Android Dialog 对话框 1.Dialog介绍 2.AlertDialog的基本使用 3.自定义对话框 Custom Dialog 一.Dialog介绍 Dialog也是Android中常用的用户界面元素,他同Menu一样也不是View的子类.让我们看一下它的继承关系: 这里要留意一下他的直接子类 AlertDialog,和间接子类 DatePickerDialog,ProgressDialog,TimePickerDialog,其中后三个我们在前面的章节已经讲过,今天我们把重点放在

Android 事件分发详解及示例代码_Android

事件分发是Android中非常重要的机制,是用户与界面交互的基础.这篇文章将通过示例打印出的Log,绘制出事件分发的流程图,让大家更容易的去理解Android的事件分发机制. 一.必要的基础知识 1.相关方法 Android中与事件分发相关的方法主要包括dispatchTouchEvent.onInterceptTouchEvent.onTouchEvent三个方法,而事件分发一般会经过三种容器,分别为Activity.ViewGroup.View.下表对这三种容器分别拥有的事件分发相关方法进行

Android ListView组件详解及示例代码_Android

Android 列表组件 ListView 列表组件是开发中经常用到组件,使用该组件在使用时需要为它提供适配器,由适配器提供来确定显示样式和显示数据. 下面看一个例子: 新建一个项目Lesson8_ListViewTest,Activity name是MainListViewTest . MainListViewTest.java的代码是: package android.basic.lesson8; import android.app.Activity; import android.os.

《Android程序设计》一1.5 示例代码

1.5 示例代码 Android SDK安装完成并经过测试证明已经可以正常工作之后,可以进一步探索Android.即使你对Android的框架不熟悉,并且是Java新手,学习一些示例代码可以使你在阅读本书的其他部分之前,对SDK的安装更有自信. 1.5.1 SDK示例代码 SDK提供了最便捷的示例代码.可以使用SDK示例创建新的项目,如图1-16所示.选定的示例会出现在Eclipse窗口的左窗格,可以浏览该示例中所包含的文件,运行它查看它实现了什么功能.如果已经能够熟练使用IDE来调试这些代码,

需要安全认证的远程EJB调用示例(Jboss EAP 6.2环境)

一,Remote EJB 服务接口定义: 1 package yjmyzz.ejb.server.helloworld; 2 3 public interface HelloWorldService { 4 5 public String sayHello(String name); 6 7 } 实现: 1 package yjmyzz.ejb.server.helloworld; 2 3 import javax.annotation.security.RolesAllowed; 4 impo

Android Content Provider详解及示例代码_Android

Android:Content Provider的使用. 1.Content Provider 简介 2.使用现成的Content Provider 3.定义自己的Content Provider 一.Content Provider 简介 我们说Android应用程序的四个核心组件是:Activity.Service.Broadcast Receiver 和 Content Provider.在Android中,应用程序彼此之间相互独立的,它们都运行在自己独立的虚拟机中.Content Pro