android SDK提供了Service,用于类似*nix守护进程或者windows的服务。

Service有两种类型:

  1. 本地服务(Local Service):用于应用程序内部
  2. 远程服务(Remote Sercie):用于android系统内部的应用程序之间

前者用于实现应用程序自己的一些耗时任务,比如查询升级信息,并不占用应用程序比如Activity所属线程,而是单开线程后台执行,这样用户体验比较好。

后者可被其他应用程序复用,比如天气预报服务,其他应用程序不需要再写这样的服务,调用已有的即可。

编写不需和Activity交互的本地服务示例

本地服务编写比较简单。首先,要创建一个Service类,该类继承android的Service类。这里写了一个计数服务的类,每秒钟为计数器加一。在服务类的内部,还创建了一个线程,用于实现后台执行上述业务逻辑。

 原文转自(含原码):http://yangguangfu.iteye.com/blog/699306
 

package com.easymorse;

import
android.app.Service;
import
android.content.Intent;
import
android.os.IBinder;
import
android.util.Log;

public

class
CountService
extends
Service {

private

boolean
threadDisable;

private

int
count;

@Override
   
public
IBinder onBind(Intent intent) {
       
return

null
;
    }

@Override
   
public

void
onCreate() {
       
super
.onCreate();
       
new
Thread(
new
Runnable() {

@Override
           
public

void
run() {
               
while
(
!
threadDisable) {
                   
try
{
                        Thread.sleep(
1000
);
                    }
catch
(InterruptedException e) {
                    }
                    count
++
;
                    Log.v(
"
CountService
"
,
"
Count is
"

+
count);
                }
            }
        }).start();
    }

@Override
   
public

void
onDestroy() {
       
super
.onDestroy();
       
this
.threadDisable
=

true
;
        Log.v(
"
CountService
"
,
"
on destroy
"
);
    }

public

int
getCount() {
       
return
count;
    }

}

需要将该服务注册到配置文件AndroidManifest.xml中,否则无法找到:

<?
xml version="1.0" encoding="utf-8"
?>

<
manifest
xmlns:android
="http://schemas.android.com/apk/res/android"

    package
="com.easymorse"
android:versionCode
="1"
android:versionName
="1.0"
>

   
<
application
android:icon
="@drawable/icon"
android:label
="@string/app_name"
>

       
<
activity
android:name
=".LocalServiceDemoActivity"

            android:label
="@string/app_name"
>

           
<
intent-filter
>

               
<
action
android:name
="android.intent.action.MAIN"

/>

               
<
category
android:name
="android.intent.category.LAUNCHER"

/>

           
</
intent-filter
>

       
</
activity
>

       
<
service
android:name
="CountService"

/>

   
</
application
>

   
<
uses-sdk
android:minSdkVersion
="3"

/>

</
manifest/>

 

在Activity中启动和关闭本地服务。

package
com.easymorse;

import
android.app.Activity;
import
android.content.Intent;
import
android.os.Bundle;

public

class
LocalServiceDemoActivity
extends
Activity {
   
/**
Called when the activity is first created.
*/

    @Override
   
public

void
onCreate(Bundle savedInstanceState) {
       
super
.onCreate(savedInstanceState);
        setContentView(R.layout.main);

this
.startService(
new
Intent(
this
, CountService.
class
));
    }

@Override
   
protected

void
onDestroy() {
       
super
.onDestroy();
       
this
.stopService(
new
Intent(
this
, CountService.
class
));
    }
}

 

可通过日志查看到后台线程打印的计数内容。

编写本地服务和Activity交互的示例

上面的示例是通过startService和stopService启动关闭服务的。适用于服务和activity之间没有调用交互的情况。如果之间需要传递参数或者方法调用。需要使用bind和unbind方法。

具体做法是,服务类需要增加接口,比如ICountService,另外,服务类需要有一个内部类,这样可以方便访问外部类的封装数据,这个内部类
需要继承Binder类并实现ICountService接口。还有,就是要实现Service的onBind方法,不能只传回一个null了。

这是新建立的接口代码:

package
com.easymorse;

public

interface
ICountService {
   
public

abstract

int
getCount();
}

 

修改后的CountService代码:

package
com.easymorse;

import
android.app.Service;
import
android.content.Intent;
import
android.os.Binder;
import
android.os.IBinder;
import
android.util.Log;

public

class
CountService
extends
Service
implements
ICountService {

private

boolean
threadDisable;

private

int
count;

private
ServiceBinder serviceBinder
=
new
ServiceBinder();

public

class
ServiceBinder
extends
Binder
implements
ICountService{
        @Override
       
public

int
getCount() {
           
return
count;

}

}

@Override

public
IBinder onBind(Intent intent) {
       
return
serviceBinder;
    }

@Override
   
public

void
onCreate() {
       
super
.onCreate();
       
new
Thread(
new
Runnable() {

@Override
           
public

void
run() {
               
while
(
!
threadDisable) {
                   
try
{
                        Thread.sleep(
1000
);
                    }
catch
(InterruptedException e) {
                    }
                    count
++
;
                    Log.v(
"
CountService
"
,
"
Count is
"

+
count);
                }
            }
        }).start();
    }

@Override
   
public

void
onDestroy() {
       
super
.onDestroy();
       
this
.threadDisable
=

true
;
        Log.v(
"
CountService
"
,
"
on destroy
"
);
    }

/*
(non-Javadoc)
     * @see com.easymorse.ICountService#getCount()
    
*/

   
public

int
getCount() {
       
return
count;
    }

}

 

服务的注册也要做改动,AndroidManifest.xml文件:

<?
xml version="1.0" encoding="utf-8"
?>

<
manifest
xmlns:android
="http://schemas.android.com/apk/res/android"

    package
="com.easymorse"
android:versionCode
="1"
android:versionName
="1.0"
>

   
<
application
android:icon
="@drawable/icon"
android:label
="@string/app_name"
>

       
<
activity
android:name
=".LocalServiceDemoActivity"

            android:label
="@string/app_name"
>

           
<
intent-filter
>

               
<
action
android:name
="android.intent.action.MAIN"

/>

               
<
category
android:name
="android.intent.category.LAUNCHER"

/>

           
</
intent-filter
>

       
</
activity
>

       
<
service
android:name
="CountService"
>

           
<
intent-filter
>

               
<
action
android:name
="com.easymorse.CountService"
/>

           
</
intent-filter
>

       
</
service
>

   
</
application
>

   
<
uses-sdk
android:minSdkVersion
="3"

/>

</
manifest
>

 

Acitity代码不再通过startSerivce和stopService启动关闭服务,另外,需要通过ServiceConnection的内部类实现来连接Service和Activity。

package
com.easymorse;

import
android.app.Activity;
import
android.content.ComponentName;
import
android.content.Intent;
import
android.content.ServiceConnection;
import
android.os.Bundle;
import
android.os.IBinder;
import
android.util.Log;

public

class
LocalServiceDemoActivity
extends
Activity {

private
ServiceConnection serviceConnection
=

new
ServiceConnection() {

@Override
       
public

void
onServiceConnected(ComponentName name, IBinder service) {
            countService
=
(ICountService) service;
            Log.v(
"
CountService
"
,
"
on serivce connected, count is
"

                   
+
countService.getCount());
        }

@Override
       
public

void
onServiceDisconnected(ComponentName name) {
            countService
=

null
;
        }

};

private
ICountService countService;

/**
Called when the activity is first created.
*/

    @Override
   
public

void
onCreate(Bundle savedInstanceState) {
       
super
.onCreate(savedInstanceState);
        setContentView(R.layout.main);
       
this
.bindService(
new
Intent(
"
com.easymorse.CountService
"
),
               
this
.serviceConnection, BIND_AUTO_CREATE);
    }

@Override
   
protected

void
onDestroy() {

this
.unbindService(serviceConnection);

super
.onDestroy();       //注意先后

}
}

 

编写传递基本型数据的远程服务

上面的示例,可以扩展为,让其他应用程序复用该服务。这样的服务叫远程(remote)服务,实际上是进程间通信(RPC)。

这时需要使用android接口描述语言(AIDL)来定义远程服务的接口,而不是上述那样简单的java接口。扩展名为aidl而不是java。
可用上面的ICountService改动而成ICountSerivde.aidl,eclipse会自动生成相关的java文件。

package com.easymorse;

interface
ICountService {
   
int
getCount();
}

 

编写服务(Service)类,稍有差别,主要在binder是通过远程获得的,需要通过桩(Stub)来获取。桩对象是远程对象的本地代理。

package
com.easymorse;

import
android.app.Service;
import
android.content.Intent;
import
android.os.IBinder;
import
android.os.RemoteException;
import
android.util.Log;

public

class
CountService
extends
Service {

private

boolean
threadDisable;

private

int
count;

private
ICountService.Stub serviceBinder
=

new
ICountService.Stub() {

@Override
       
public

int
getCount()
throws
RemoteException {
           
return
count;
        }
    };

@Override
   
public
IBinder onBind(Intent intent) {
       
return
serviceBinder;
    }

@Override
   
public

void
onCreate() {
       
super
.onCreate();
       
new
Thread(
new
Runnable() {

@Override
           
public

void
run() {
               
while
(
!
threadDisable) {
                   
try
{
                        Thread.sleep(
1000
);
                    }
catch
(InterruptedException e) {
                    }
                    count
++
;
                    Log.v(
"
CountService
"
,
"
Count is
"

+
count);
                }
            }
        }).start();
    }

@Override
   
public

void
onDestroy() {
       
super
.onDestroy();
       
this
.threadDisable
=

true
;
        Log.v(
"
CountService
"
,
"
on destroy
"
);
    }
}

 

配置文件AndroidManifest.xml和上面的类似,没有区别。

在Activity中使用服务的差别不大,只需要对ServiceConnection中的调用远程服务的方法时,要捕获异常。

private
ServiceConnection serviceConnection
=

new
ServiceConnection() {

@Override
   
public

void
onServiceConnected(ComponentName name, IBinder service) {
        countService
=
(ICountService) service;
       
try
{
            Log.v(
"
CountService
"
,
"
on serivce connected, count is
"

                   
+
countService.getCount());
        }
catch
(RemoteException e) {
           
throw

new
RuntimeException(e);
        }
    }

@Override
   
public

void
onServiceDisconnected(ComponentName name) {
        countService
=

null
;
    }

};

 

这样就可以在同一个应用程序中使用远程服务的方式和自己定义的服务交互了。

如果是另外的应用程序使用远程服务,需要做的是复制上面的aidl文件和相应的包构到应用程序中,其他调用等都一样。

编写传递复杂数据类型的远程服务

远程服务往往不只是传递java基本数据类型。这时需要注意android的一些限制和规定:

  1. android支持String和CharSequence
  2. 如果需要在aidl中使用其他aidl接口类型,需要import,即使是在相同包结构下;
  3. android允许传递实现Parcelable接口的类,需要import;
  4. android支持集合接口类型List和Map,但是有一些限制,元素必须是基本型或者上述三种情况,不需要import集合接口类,但是需要对元素涉及到的类型import;
  5. 非基本数据类型,也不是String和CharSequence类型的,需要有方向指示,包括in、out和inout,in表示由客户端设置,out表示由服务端设置,inout是两者均可设置。

这里将前面的例子中返回的int数据改为复杂数据类型:

package
com.easymorse;

import
android.os.Parcel;
import
android.os.Parcelable;

public

class
CountBean
implements
Parcelable {

public

static

final
Parcelable.Creator
<
CountBean
>
CREATOR
=

new
Creator
<
CountBean
>
() {

@Override
       
public
CountBean createFromParcel(Parcel source) {
            CountBean bean
=

new
CountBean();
            bean.count
=
source.readInt();
           
return
bean;
        }

@Override
       
public
CountBean[] newArray(
int
size) {
           
return

new
CountBean[size];
        }

};

public

int
count;

@Override
   
public

void
writeToParcel(Parcel dest,
int
flags) {
        dest.writeInt(
this
.count);
    }

@Override
   
public

int
describeContents() {
       
return

0
;
    }

}

 

然后,需要在相同包下建一个同名的aidl文件,用于android生成相应的辅助文件:

package
com.easymorse;

parcelable CountBean;

 

这一步是android 1.5后的变化,无法通过adt生成aidl,也不能用一个比如全局的project.aidl文件,具体见:

http://www.anddev.org/viewtopic.php?p=20991

然后,需要在服务的aidl文件中修改如下:

package
com.easymorse;

import
com.easymorse.CountBean;

interface
ICountService {
    CountBean getCount();
}

本地/远程Service 和Activity 的交方式(转)的更多相关文章

  1. Service与Activity通信 回调方式***

    要实现service与activity的高强度通信用什么方法? service与activity之前的通信方式有很多,回调接口方式.观察者模式.广播.还有handler等,方法有很多,但要高强度地通信 ...

  2. android开发之使用Messenger实现service与activity交互

    service与activity交互的方式有多种,这里说说使用Messenger来实现两者之间的交互. Service程序 public class MessengerService extends ...

  3. android service 本地 远程 总结

    android编写Service入门 android SDK提供了Service,用于类似*nix守护进程或者windows的服务. Service有两种类型: 本地服务(Local Service) ...

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

    当一个Service在androidManifest中被声明为 process=":remote", 或者是还有一个应用程序中的Service时,即为远程Service, 远程的意 ...

  5. Android Service与Activity之间通信的几种方式

    在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务,所以在我们实际开发中,就会常常遇到Activity与Service之间的通信,我们一般在Activ ...

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

    在Android中,Service是运行在主线程中的,如果在Service中处理一些耗时的操作,就会导致程序出现ANR. 但如果将本地的Service转换成一个远程的Service,就不会出现这样的问 ...

  7. Service与Activity与交流AIDL

    深圳旅游月.终于回来了,做了很多个月,这些天来的东西会慢慢总结出来的.今天,我们正在谈论的Service小东西:沟通. 固定通信的做法比较,基本上按照写模板可以实现. 1.Service与Activi ...

  8. 一个简单的demo学习Android远程Service(AIDL的使用)

    这是milo很早之前写在论坛上的一个帖子,现在整理出来,milo也复习一下一般来说Android 的四大组件都是运行在同一个进程中的,但远程Service运行在不同的进程里.这进程间的通信是使用了An ...

  9. Android—Service与Activity的交互

    service-Android的四大组件之一.人称"后台服务"指其本身的运行并不依赖于用户可视的UI界面 实际开发中我们经常需要service和activity之间可以相互传递数据 ...

随机推荐

  1. 使用委托解决"线程间操作无效: 从不是创建控件“textBox1”的线程访问它" 问题

    在winform编程中常遇到此类问题,造成辅助线程无法给控件赋值 //定义委托 private delegate void SetTextCallback(string text); //在给text ...

  2. 经典面试题(二)附答案 算法+数据结构+代码 微软Microsoft、谷歌Google、百度、腾讯

    1.正整数序列Q中的每个元素都至少能被正整数a和b中的一个整除,现给定a和b,需要计算出Q中的前几项, 例如,当a=3,b=5,N=6时,序列为3,5,6,9,10,12 (1).设计一个函数void ...

  3. KnockOutJS学习系列----(一)

    原文地址:http://www.cnblogs.com/n-pei/archive/2011/12/23/2299217.html 好几个月没去写博客了,最近也是因为项目紧张,不过这个不是借口,J. ...

  4. This function or variable may be unsafe. Consider using scanf_s instead.

    去掉安全检查,开头加上即可: #define _CRT_SECURE_NO_WARNINGS 或者: 严重性代码 说明项目文件行禁止显示状态 错误C4996 'scanf': This functio ...

  5. Iphone JS时间

    var end_time = new Date(time).getTime();//月份是实际月份-1  var start_time= new Date(serverTime).getTime(); ...

  6. 后缀数组da3模板

    在做poj2406的时候...按论文给的rmq模板会超内存...然后网上找了http://blog.csdn.net/libin56842/article/details/46310425这位大爷的d ...

  7. 2014 ACM省赛总结

    今年ACM省赛已经过去一个星期左右了,2年的ACM训练是该做个总结了,因为前几日去參加蓝桥杯总决赛,所以没来的及写总结,如今在这小小总结一下吧-- 依晰记得去年省赛时候的样子,如今感觉那时像是个无知的 ...

  8. 依赖注入及AOP简述(三)——依赖注入的原理

    3.     “依赖注入”登场 于是诸多优秀的IT工程师开始想出了更加轻量便利.更加具有可测试性和可维护性的设计模式——IoC模式.IoC,即Inversion of Control的缩写,中文里被称 ...

  9. 不使用TNS直连数据库的三种方式

    1.在当前目录下新建tnsnames.ora文件 如windows环境下,在C:\Users\Administrator目录下新建tnsnames.ora文件,内容如下:test =(descript ...

  10. 响应式布局之BootStrap

    本文在于巩固基础 学习bootStrap官网http://getbootstrap.com/ 响应式布局是Ethan Marcotte在2010年5月份提出的一个概念,简而言之,就是一个网站能够兼容多 ...