Android中远程Service浅析

上一篇文章中简单的写了一下关于Android中Service的两种启动方式,不过都是本地的服务,今天就简单的写下关于Android中远程Service的使用,学习之前先了解两个概念,AIDL( Android Interface definition language)字面上的意思就是借口定义语言,专业一点理解就是Android进程之间通信的借口描述语言。IPC(Inter-Process Conmmunication)内部进程之间的通信,同一个手机上,如果你的APP需要访问调用另外一个APP的服务,通信的方式就是IPC。

同一个APP中Service调用

跟上篇文章不同,这次先自行创建一个名称为BookAIDLService.aidl的AIDL文件:

package com.remote.service;

interface BookAIDLService {
    int  sum(int a,int b);
}

  吐槽一下,网上很多都是这么写的,自己新增的时候没有找到如何新建一个AIDL文件,你首先需要建一个BookAIDLService.java文件,然后修改后缀名为aidl,这个时候看到效果如下:

保存之后,会自动的在gen目录下生成一个BookAIDLService.java文件,还是跟最开始一样,看下应用程序页面:

本地事件针对的是第三个按钮,先来重写下BookService:

public class BookService extends Service {

	private String tag = "BookService";
    BookAIDLService.Stub bookAIDLBinderStub=new Stub() {

		@Override
		public int sum(int a, int b) throws RemoteException {
			// TODO Auto-generated method stub
			return a+b;
		}
	};
	@Override
	public void onCreate() {
		// TODO Auto-generated method stub
		super.onCreate();
		Log.i(tag, "开始onCreate启动了");
		 Log.i("BookService","BookService的ID:"+Process.myPid());
//		try {
//			Thread.sleep(40000);
//		} catch (InterruptedException e) {
//			// TODO Auto-generated catch block
//			e.printStackTrace();
//		}

	}

	@Override
	public int onStartCommand(Intent intent, int flags, int startId) {
		// TODO Auto-generated method stub
		Log.i(tag, "开始执行onStartCommand启动了");
		Toast.makeText(this, "BookService开始了", Toast.LENGTH_SHORT).show();
		return super.onStartCommand(intent, flags, startId);

	}

	@Override
	public void onDestroy() {
		// TODO Auto-generated method stub
		Log.i(tag, "销毁onDestroy启动了");
		super.onDestroy();
	}

	@Override
	public IBinder onBind(Intent intent) {
		// TODO Auto-generated method stub
		Log.i(tag, "绑定onBind启动了");
		return bookAIDLBinderStub;
	}

	@Override
	public boolean onUnbind(Intent intent) {
		// TODO Auto-generated method stub
		Log.i(tag, "解绑onUnbind启动了");
		return super.onUnbind(intent);
	}

	class BookBinder extends Binder {
		public BookService getCurrentService() {
			return BookService.this;
		}
	}

}

  跟之前最大的不同就是在onBind方法中返回一个bookAIDLBinderStub,同时上次写的BookConnection也要冲洗写一下:

    class BookServiceConnection implements ServiceConnection{

		private BookAIDLService bookAIDLService;

		public BookServiceConnection() {
			super();
			// TODO Auto-generated constructor stub
		}

		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			// TODO Auto-generated method stub
			//获取实例
			//BookService  bookService=((BookService.BookBinder)service).getCurrentService();
			//just  do  wo  you  want to do
			 BookAIDLService	  bookAIDLService=BookAIDLService.Stub.asInterface(service);
			  try {
				int result=bookAIDLService.sum(10, 100);
				Log.i("BookService", "BookAIDLService调用结果:"+result);
			} catch (RemoteException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}

		}

		@Override
		public void onServiceDisconnected(ComponentName name) {
			// TODO Auto-generated method stub

		}

    }
}

  前台的调用是:

Intent binderStartIntent=new Intent("com.example.googleservice.BookService.AIDL");
 			 connection=new BookServiceConnection();
			bindService(binderStartIntent, connection,Context.BIND_AUTO_CREATE);

  这里Intent是隐式调用,如果不是很熟悉可以参考我之前的文章,AndroidManifest.xml文件中需要重新改动一下:

       <service android:name="com.example.googleservice.BookService"
            android:process=":remote">
            <intent-filter>
               <action android:name="com.example.googleservice.BookService.AIDL"/>
            </intent-filter>
        </service>

  调用结果如下:

不同的App之间的调用

不同之间的调用,由于相互之间要相互通信,同样的需要定义与服务端的aidl名相同的aidl,新建一个Android项目,然后结构如下:

客户端页面就不用写了,就一个调用按钮,客户端把服务端的BookConnection拷贝过来:

   class BookServiceConnection implements ServiceConnection{

		private BookAIDLService bookAIDLService;

		public BookServiceConnection() {
			super();
			// TODO Auto-generated constructor stub
		}

		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			// TODO Auto-generated method stub
			//获取实例
			//BookService  bookService=((BookService.BookBinder)service).getCurrentService();
			//just  do  wo  you  want to do
			 BookAIDLService	  bookAIDLService=BookAIDLService.Stub.asInterface(service);
			  try {
				int result=bookAIDLService.sum(10, 100);
				Log.i("BookService", "客户端BookAIDLService调用结果:"+result);
			} catch (RemoteException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}

		}

		@Override
		public void onServiceDisconnected(ComponentName name) {
			// TODO Auto-generated method stub

		}

    }

  客户端调用:

		Intent binderStartIntent=new Intent("com.example.googleservice.BookService.AIDL");
 			 connection=new BookServiceConnection();
			bindService(binderStartIntent, connection,Context.BIND_AUTO_CREATE);

  调用结果如下:

好了,至此简单的讲了一下Android中的远程服务调用,很多概念没有讲,不会掉书袋,有兴趣可以自己私下了解下,不同的进程之间传递数据,Android对这类数据的格式支持是非常有限,基本上只能传递Java的基本数据类型、字符串、List或Map,如果想传一个自定义的类,必须要让这个类去实现Parcelable接口,并且要给这个类也定义一个同名的AIDL文件。大同小异,各位可以自行研究,一不小心又周五了,哎~

时间: 2024-10-13 16:22:38

Android中远程Service浅析的相关文章

android 远程Service以及AIDL的跨进程通信

在Android中,Service是运行在主线程中的,如果在Service中处理一些耗时的操作,就会导致程序出现ANR. 但如果将本地的Service转换成一个远程的Service,就不会出现这样的问题了. 转换成远程Service非常简单,只需要在注册Service的时候将他的android:process的属性制定成 :remote就可以了. 重新运行项目,你会发现,不会出现ANR了. 为什么将MyService转换成远程Service后就不会导致程序ANR了呢?这是由于,使用了远程Serv

android的Service的实例

package com.android.service; import android.app.IntentService;import android.content.Intent; public class HelloIntentService extends IntentService{ public HelloIntentService() {        super("HelloIntentService");        // TODO Auto-generated c

Android源码浅析(一)——VMware Workstation Pro和Ubuntu Kylin 16.04 LTS安装配置

Android源码浅析(一)--VMware Workstation Pro和Ubuntu Kylin 16.04 LTS安装配置 最近地方工作,就是接触源码的东西了,所以好东西还是要分享,系列开了这么多,完结 的也没几个,主要还是自己覆盖的太广了,却又不精通,嘿嘿,工作需要,所以写下了本篇博客 一.VMware 12 我选择的虚拟机试VMware,挺好用的感觉,下载VMware就不说了,善用搜索键嘛,这里我提供一个我现在在用的 下载地址:链接:http://pan.baidu.com/s/1k

(六)Android中Service通信

一.启动Service并传递参数 传递参数时只需在startService启动的Intent中传入数据便可,接收参数时可在onStartCommand函数中通过读取第一个参数Intent的内容来实现 1.MainActivity.java package com.example.shiyanshi.serviceconnected; import android.app.Activity;import android.content.Intent;import android.os.Bundle

Android 测试Service的生命周期

1 package com.example.myapp4; 2 3 import android.support.v7.app.ActionBarActivity; 4 import android.content.Intent; 5 import android.os.Bundle; 6 import android.view.Menu; 7 import android.view.MenuItem; 8 import android.view.View; 9 import android.w

Android之Service通信-(2)

一.Service通过IBinder与Activity进行通信 在Service中进行下载 Service package chuiyuan.lsj.androidjava.service; import android.app.Service; import android.content.Intent; import android.os.Binder; import android.os.IBinder; import android.util.Log; import android.wi

Android——判断Service是否已经启动

延续百度地图定位的Demo,采用Service来进行百度定位,并且将数据上传到服务器上遇到了一个问题:在真机中使用清理内存来关闭程序的之后,Service会被关闭,但是过几秒中,它又会自动重启:重启就算了,而且再次登陆系统的时候,又会开启一个一样的服务,在LogCat中就会看到每次都获取到两次的定位数据.然后想想是否可以在建立Service之前判断这个服务有没有被创建?只要能做这个判断,那么服务存在我们就不管它,如果不存在则创建,本着这个思路,百度发现可行(Service后台服务创建时最好都要判

Android Web Service学习总结(一)

最近学习android平台调用webWebService,学习了一篇不错的博客(http://blog.csdn.net/lyq8479/article/details/6428288),可惜是2011年时的方法,而不适合现在android4.0之后的android版本,所以通过一番学习和研究,总结如下. web Service简介 通俗的理解:通过使用WebService,我们能够像调用本地方法一样去调用远程服务器上的方法.我们并不需要关心远程的那个方法是Java写的,还是PHP或C#写的:我

如何从python代码中直接访问Android的Service

在Kivy中,通过pyjnius扩展可以间接调用Java代码,而pyjnius利用的是Java的反射机制.但是在Python对象和Java对象中转来转去总让人感觉到十分别扭.好在android提供了binder这个进程间通信的功能,Java中的Service也是基于Binder的C++代码封装来实现进程间通信的,这也为从Python代码中绕开pyjnius直接访问Java代码提供了可能,既然Java的Service是基于C++的封装来实现的,也同样可以在Python中封装同样的C++代码,这篇文