码迷,mamicode.com
首页 > 移动开发 > 详细

Android Activity与远程Service的通信学习总结

时间:2015-08-21 15:40:04      阅读:357      评论:0      收藏:0      [点我收藏+]

标签:远程service

当一个Service在androidManifest中被声明为 process=":remote", 或者是另一个应用程序中的Service时,即为远程Service, 远程的意思是和当前程序不在同一个进程中运行。Activity和远程Service的跨进程通信(IPC)通过Binder机制,使用AIDL服务实现。

而常见的本地Service与activity的交互通过bindService即可实现,见 http://blog.csdn.net/guige_csdn/article/details/47809895


一.  定义远程服务端


1.新建一个工程,工程目录如下

技术分享技术分享


2. 文件内容

aidl传递复杂对象时,需要该对象实现Parcelable或Serializable接口,并在aidl文件中声明


Student.java

package com.example.aidl;

import java.util.Locale;

import android.R.integer;
import android.os.Parcel;
import android.os.Parcelable;

public class Student implements Parcelable {

	public static final int SEX_MALE = 1;  
    public static final int SEX_FEMALE = 2; 

	public int sno;
	public String name;
	public int sex;
	public int age;

	public Student() {
	}

	private Student(Parcel in) {
		readFromParcel(in);
	}

	public void readFromParcel(Parcel in) {
		sno = in.readInt();
		name = in.readString();
		sex = in.readInt();
		age = in.readInt();
	}

	@Override
	public void writeToParcel(Parcel dest, int flags) {
       dest.writeInt(sno);
       dest.writeString(name);
       dest.writeInt(sex);
       dest.writeInt(age);
	}

	public static final Parcelable.Creator<Student> CREATOR = new Parcelable.Creator<Student>() {

		@Override
		public Student createFromParcel(Parcel in) {
			return new Student(in);
		}

		@Override
		public Student[] newArray(int size) {
			return new Student[size];
		}
	};

	@Override
	public int describeContents() {
		return 0;
	}
	
	@Override  
    public String toString() {  
        return String.format(Locale.ENGLISH, "Student[ %d, %s, %d, %d ]", sno, name, sex, age);  
    } 

}


Student.aidl

<pre name="code" class="java">package com.example.aidl;

parcelable Student;



IService.aidl

package com.example.aidl;

import com.example.aidl.Student;

 interface IMyService{
 
    List<Student> getStudent();
    void addStudent(in Student student); //in标记为输入类型
  }


然后再写远程服务类StudentService.java

package com.example.remote_service;

import java.util.LinkedList;
import java.util.List;

import com.example.aidl.IMyService;
import com.example.aidl.Student;

import android.R.integer;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.Parcel;
import android.os.RemoteException;
import android.util.Log;

public class StudentService extends Service {

	private static final String TAG = "StudentService";
	
	private static final String PACKAGE_PEMISSION = "com.yjq.service";  
	
	private LinkedList<Student> studentList = new LinkedList<Student>();
		
	@Override
	public IBinder onBind(Intent intent) {
		Log.d(TAG, String.format("on bind,intent = %s", intent.toString()));  
		return mBinder;
	}

	   //继承由AIDL生成的抽象类 
//	实现aidl里的接口的方法 ,以供调用
	//这里实现了aidl中的抽象函数  
    private final IMyService.Stub mBinder = new IMyService.Stub() { 

		@Override
		public List<Student> getStudent() throws RemoteException {
			return studentList;
		}

		@Override
		public void addStudent(Student student) throws RemoteException {
			if (student != null) {
				synchronized (studentList) {
					studentList.add(student);
				}
			}
		}		
		
		//在这里可以做权限认证,return false意味着客户端的调用就会失败,比如下面,只允许包名为 PACKAGE_SAYHI 的客户端通过,  
        //其他apk将无法完成调用过程  
		@Override
        public boolean onTransact(int code, Parcel data, Parcel reply, int flags)  
                throws RemoteException {  
            String packageName = null;  
            String[] packages = StudentService.this.getPackageManager().  
                    getPackagesForUid(getCallingUid());  
            if (packages != null && packages.length > 0) {  
                packageName = packages[0];  
            }  
            Log.d(TAG, "onTransact: " + packageName);  
            if (!PACKAGE_PEMISSION.equals(packageName)) {  
                return false;  
            }  
            return super.onTransact(code, data, reply, flags);  
        }  
		
	};

	@Override
	public void onCreate() {
		super.onCreate();
	    Log.d(TAG, "student服务创建");
	    
	    synchronized (studentList) {  
            for (int i = 1; i < 6; i++) {  
                Student student = new Student();  
                student.name = "student#" + i;  
                student.age = i * 5;  
                studentList.add(student);  
            }  
        }  
	    
	}

	
	@Override
	public int onStartCommand(Intent intent, int flags, int startId) {
                Log.d(TAG, "student服务启动");
		return super.onStartCommand(intent, flags, startId);
	}

	@Override
	public void onDestroy() {
		Log.d(TAG, "student服务结束");
		super.onDestroy();
	}

	
}



Stub这个内部类是一个实现了IMyService接口的Binder抽象类

onTransact()是通过应用t程序的包名来限制权限的,通过验证的程序才能绑定。 允许绑定的客户端程序 的包名,可在客户端的androidManifist中查看。


并注意在androidManifest中声明

        <service  
            android:name="com.example.remote_service.StudentService"
            android:process=":remote"
            >
            <intent-filter >
                <!-- 这里的action是隐式bindService时用到的  -->
                <action  android:name="com.example.remoteservice" />
                <category android:name="android.intent.category.DEFAULT" /> 
            </intent-filter>
            
        </service>
要声明action以供隐式bindService

二. 本地Activity如何绑定远程Service

新建一个项目作为绑定远程Service的客户端,将上面的aidl所在包复制到src中,注意整个包名和文件名都要完全一样,否则无法远程绑定

再写MainActivity.java来进行绑定操作

</pre><p></p><p><pre name="code" class="java">package com.yjq.local_service;

import java.util.List;

import com.example.aidl.IMyService;
import com.example.aidl.Student;
import com.yjq.local_service.MyService.MyBinder;
import com.yjq.service.R;
import android.os.Bundle;
import android.os.Handler;
import android.os.IBinder;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;

public class MainActivity extends Activity {

	private static final String TAG = "MainActivity_local";

//绑定远程Service的action
	public static final String BIND_REMOTE_SERVICE_ACTION = "com.example.remoteservice";

	// 远程服务 , 服务的类是在aidl中定义的
	private IMyService mRemoteService;

	private Button bindRemoteServiceButton;

	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		bindRemoteServiceButton =(Button)findViewById(R.id.bind_remote_service);

                              bindRemoteServiceButton.setOnClickListener( new OnClickListener() {
			@Override
			public void onClick(View v) {
				bindRemoteService();
			}
		});
	}


	/**
	 * 创建并绑定远程Service
	 * */
	private void bindRemoteService() {
		Intent intentService = new Intent(BIND_REMOTE_SERVICE_ACTION);
		intentService.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
		bindService(intentService, remoteServiceConnection, BIND_AUTO_CREATE);
	}

	
	

	/**
	 * 绑定远程服务的ServiceConnection类
	 */
	ServiceConnection remoteServiceConnection = new ServiceConnection() {

		@Override
		public void onServiceDisconnected(ComponentName name) {
			mRemoteService = null;
		}

		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			mRemoteService = IMyService.Stub.asInterface(service);

			new Thread(new Runnable() {
				@Override
				public void run() {
					try {
						List<Student> students = mRemoteService.getStudent();
						for (Student s : students) {
							System.out.println("远程调用: " + s.toString());
						}
					} catch (Exception e) {
						// TODO: handle exception
					}
				}
			}).start();

		}
	};

	@Override
	protected void onDestroy() {
		unbindService(remoteServiceConnection);
		super.onDestroy();
	}

}



activity中通过

bindService(intentService, remoteServiceConnection, BIND_AUTO_CREATE);

来绑定远程Service


实现接口的几条说明:

 *不会有返回给调用方的异常

 *默认IPC调用是同步的。如果已知IPC服务端会花费很多毫秒才能完成,那就不要在ActivityView线程中调用,否则会引起应用程序挂起(Android可能会显示“应用程序未响应”对话框),所以上面创建了一个Thread来执行

 *AIDL接口中只支持方法,不能声明静态成员。


而且,当调用结束时,应该使用

		unbindService(remoteServiceConnection);
来解除绑定,否则会造成内存泄漏


三. 运行测试 

先安装服务端的程序,再运行客户端程序,点击绑定按钮,即看到结果

08-20 22:49:48.447: I/System.out(1319): 远程调用: Student[ 0, student#1, 0, 5 ]
08-20 22:49:48.457: I/System.out(1319): 远程调用: Student[ 0, student#2, 0, 10 ]
08-20 22:49:48.457: I/System.out(1319): 远程调用: Student[ 0, student#3, 0, 15 ]
08-20 22:49:48.457: I/System.out(1319): 远程调用: Student[ 0, student#4, 0, 20 ]
08-20 22:49:48.457: I/System.out(1319): 远程调用: Student[ 0, student#5, 0, 25 ]

Demo下载: http://download.csdn.net/detail/guige_csdn/9032089


四. 在android中的应用

摘自:http://blog.csdn.net/u011485531/article/details/12073647

下面我们就讲一个挂断电话的例子,我们知道,Android的api里面是没有挂断电话的操作的

那如果我们的应用就有这样一个需求呢,那怎么办

其实也很容易,就是通过aidl来挂断电话,那么我们现在来写一下

首先,我们先把两个aidl文件复制到src目录下面

技术分享

你肯定会问,这些aidl文件是在那里找到的,那么我可以告诉你,这些是Android的源码里面找到的,就是在base这个下面的

技术分享

大家可以下载下来看看,如果有不会的,请看点击这里


既然把aidl文件拷贝过去之后,我们就可以写代码啦

com.xiaobin.endcall.MainActivity

[java] view plaincopy
  1. package com.xiaobin.endcall;  
  2.   
  3. import java.lang.reflect.Method;  
  4.   
  5. import com.android.internal.telephony.ITelephony;  
  6.   
  7. import android.app.Activity;  
  8. import android.os.Bundle;  
  9. import android.os.IBinder;  
  10. import android.view.View;  
  11.   
  12. public class MainActivity extends Activity  
  13. {  
  14.   
  15.     @Override  
  16.     public void onCreate(Bundle savedInstanceState)  
  17.     {  
  18.         super.onCreate(savedInstanceState);  
  19.         this.setContentView(R.layout.main);  
  20.     }  
  21.   
  22.     public void click(View v)  
  23.     {  
  24.         endCall();  
  25.     }  
  26.       
  27.     //挂断电话  
  28.     private void endCall()  
  29.     {  
  30.         try  
  31.         {  
  32.             //通过反射拿到android.os.ServiceManager里面的getService这个方法的对象  
  33.             Method method = Class.forName("android.os.ServiceManager").getMethod("getService", String.class);  
  34.             //通过反射调用这个getService方法,然后拿到IBinder对象,然后就可以进行aidl啦  
  35.             IBinder iBinder = (IBinder) method.invoke(nullnew Object[] {TELEPHONY_SERVICE});  
  36.             ITelephony telephony = ITelephony.Stub.asInterface(iBinder);  
  37.             telephony.endCall();  
  38.         }  
  39.         catch (Exception e)  
  40.         {  
  41.             e.printStackTrace();  
  42.         }  
  43.     }  
  44.       
  45. }  


因为那个服务没有公开的,所以我们就要用到反射来调用这个服务,来拿到Binder对象


还有重要的一点喔,要加上相应的权限,不然挂不断的喔

[html] view plaincopy
  1. <uses-permission android:name="android.permission.CALL_PHONE"/>  

好啦,到这里,挂断电话的操作就完成的啦,可以测试一下啦,然后我们手机卫士的这个项目的黑名单那里也有挂断电话的操作的喔

挂断电话的例子下载: http://download.csdn.net/detail/guige_csdn/9032239


版权声明:本文为博主原创文章,未经博主允许不得转载。

Android Activity与远程Service的通信学习总结

标签:远程service

原文地址:http://blog.csdn.net/guige_csdn/article/details/47832043

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!