综合实例2:客户端访问远程Service服务
实现:通过一个按钮来获取远程Service的状态,并显示在两个文本框中。
思路:假设A应用需要与B应用进行通信,调用B应用中的getName()、getAuthor()方法,B应用以Service方式向A应用提供服务。所以,我们可以将A应用看成是客户端,B应用为服务端,分别命名为AILDClient、AILDServer.
一、服务端应用程序
1.src/com.example.aildserver/song.aidl:AILD文件
当完成aidl文件创建后,选择保存,eclipse会自动在项目的gen目录中同步生成Song.java接口文件。接口文件中生成一个Stub抽象类,里面包括aidl定义的方法,还包括一些其它辅助性的方法,如geName()、getSong()方法,我们可以通过这两个方法实现客户端读写Service服务端数据。
- package com.example.aildserver;
- interface Song
- {
- String getName();
- String getSong();
- }
位置如下:
编写Aidl文件时,需要注意:
1.接口名和aidl文件名相同;
2.接口和方法前不用加访问权限修饰符public,private等,也不能用final,static;
3.Aidl默认支持的类型包话java基本类型(int、long、boolean等)和(String、List、Map、CharSequence),使用这些类型时不需要import声明。对于List和Map中的元素类型必须是Aidl支持的类型。如果使用自定义类型作为参数或返回值,自定义类型必须实现Parcelable接口。
4.自定义类型和AIDL生成的其它接口类型在aidl描述文件中,应该显式import,即便在该类和定义的包在同一个包中。
5.在aidl文件中所有非Java基本类型参数必须加上in、out、inout标记,以指明参数是输入参数、输出参数还是输入输出参数。
6.Java原始类型默认的标记为in,不能为其它标记。
2.src/com.example.aildserver/MyService.java
功能:Service子类,完成Service服务
开发核心步骤:
(1)重写Service的onBind()方法(用于返回一个IBinder对象)、onCreate()方法、onDestroy() 方法、onUnbind()方法;
(2)定义一个Stub的子类,该内部类实现了IBinder、Song两个接口,该子类对象将作为远程Service的onBind()方法返回IBinder对象的代理传给客户端的ServiceConnection的onServiceConnected方法的第二个参数。
- package com.example.aildserver;
- import com.example.aildserver.Song.Stub;
- import android.app.Service;
- import android.content.Intent;
- import android.os.Binder;
- import android.os.IBinder;
- import android.os.RemoteException;
- public class MyService extends Service {
- private String[] names = new String[] {"林俊杰","蔡依林","邓紫棋"};
- private String[] songs = new String[] {"可惜没如果","第三人称","多远都要在一起"};
- private String name,song;
- private int current=1; //当前位置
- private MyBinder binder = new MyBinder(); //实例化一个IBinder对象
- /*0.Stub内部类
- * 该内部类实现了IBinder、Song两个接口,这个Stub类将会作为远程Service的回调类。*/
- public class MyBinder extends Stub
- {
- //a.客户端回调该方法获取歌手名
- public String getName() throws RemoteException
- {
- return name;
- }
- //b.客户端回调该方法获取歌曲
- public String getSong() throws RemoteException
- {
- return song;
- }
- }
- /*1.onBind方法
- * service用于返回一个IBinder对象给客户端方便通信
- */
- @Override
- public IBinder onBind(Intent arg0) {
- return binder;
- }
- /*2.onCreate方法
- * 当Service启动后,自动调用该方法,用于初始化
- * */
- public void onCreate() {
- name = names[current]; //给name、song赋值
- song = songs[current];
- System.out.println("Service print:name="+name+"song="+song);
- super.onCreate();
- }
- /*3.onDestroy方法
- * 当访问者调用Context.stopService方法后,调用该方法关闭Service服务
- * */
- public void onDestroy() {
- super.onDestroy();
- }
- /*4.onUnbind方法
- * 当访问者调调用Context.unBind()方法后,调用该方法与Service解除绑定*/
- public boolean onUnbind(Intent intent) {
- return false;
- }
- }
注意1:客户端访问Service时,Android并不是直接返回Service对象给客户端,Service只是将一个回调对象(IBinder对象)通过onBind()方法回调给客户端。
注意2:与绑定本地Service不同的是,本地Service的onBind()方法会直接把IBinder对象本身传给客户端的ServiceConnection的onServiceConnected方法的第二个参数。但远程Service的onBind()方法只是将IBinder对象的代理传给客户端的ServiceConnection的onServiceConnected方法的第二个参数。当客户端获取了远程的Service的IBinder对象的代理之后,接下来可通过该IBinder对象去回调远程Service的属性或方法。
3.AndroidManifest.xml
功能:配置Service组件,并指定其action属性(方便其他应用程序启动该Service服务)
- <application
- ........
- <!-- 配置service -->
- <service android:name=".MyService">
- <intent-filter>
- <action android:name="com.jiangdongguo.service"/>
- </intent-filter>
- </service>
- </application>
二、客户端应用程序
1.拷贝服务端.aidl文件到客户端 把AIDLService应用中aidl文件所在package连同aidl文件一起拷贝到客户端AIDLClient应用,eclipse会自动在A应用的gen目录中为aidl文件同步生成Song.java接口文件,接下来就可以在AIDLClient应用中实现与AIDLService应用通信。
2.src/com.example.aildclient/MainActivity.java
功能:(1)启动服务端Service服务;(2)获取返回的IBinder代理对象,并完成与服务端程序的通信
- package com.example.aildclient;
- import com.example.aildserver.Song;
- import android.app.Activity;
- import android.app.Service;
- import android.content.ComponentName;
- import android.content.Intent;
- import android.content.ServiceConnection;
- import android.os.Bundle;
- import android.os.IBinder;
- import android.os.RemoteException;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.widget.Button;
- import android.widget.EditText;
- public class MainActivity extends Activity {
- private Button getBtn;
- private EditText song;
- private EditText name;
- private Song binder;
- //1.创建一个ServiceConnection对象
- private ServiceConnection conn = new ServiceConnection()
- {
- public void onServiceConnected(ComponentName name, IBinder service)
- {
- binder = Song.Stub.asInterface(service); //获取Service返回的代理IBinder对象
- }
- public void onServiceDisconnected(ComponentName name) {
- }
- };
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- getBtn=(Button)findViewById(R.id.get);
- song=(EditText)findViewById(R.id.song);
- name=(EditText)findViewById(R.id.name);
- //2.指定要启动的Service
- Intent intent = new Intent("com.jiangdongguo.service");
- bindService(intent, conn, Service.BIND_AUTO_CREATE);
- getBtn.setOnClickListener(new OnClickListener(){
- public void onClick(View arg0)
- {
- try {
- name.setText(binder.getName());
- song.setText(binder.getSong());
- } catch (RemoteException e) {
- e.printStackTrace();
- }
- }
- });
- }
- }
对于远程服务调用,远程服务返回给客户端的对象为代理对象,客户端在onServiceConnected(ComponentName name, IBinder service)方法引用该对象时不能直接强转成接口类型的实例,而应该使用asInterface(IBinder iBinder)进行类型转换。
三、效果演示