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. MYSQL 加密的 3 类方法

    背景: 若你想要储存一些由可能包含任意字节值的加密函数返回的结果,使用BLOB列而不是 CHAR 或VARCHAR 列,从而避免由于结尾空格的删除而改变一些数据值的潜在问题. 这一句话来自官方文件,记 ...

  2. 抛弃QP

        随着软件的慢慢进行发现QP根本无法建立多个实例,也就是无法在多个任务中同时使用QP的事件回调 架构,这点同libevent不同,最终决定放弃之,乖乖的用freeRTOS多任务方案,workin ...

  3. 要熟悉QT的所有类和元类系统,当然还有qmake

    http://doc.qt.io/qt-5/classes.html http://doc.qt.io/qt-5/gettingstarted.html http://doc.qt.io/qt-5/q ...

  4. System.Web.Caching.Cache类 缓存 各种缓存依赖

    原文:System.Web.Caching.Cache类 缓存 各种缓存依赖 Cache类,是一个用于缓存常用信息的类.HttpRuntime.Cache以及HttpContext.Current.C ...

  5. rails跑通第一个demo

    rails -h 查看帮助 Usage: rails new APP_PATH [options] Options: -r, [--ruby=PATH] # Path to the Ruby bina ...

  6. JavaEE Tutorials (25) - 使用Java EE拦截器

    25.1拦截器概述380 25.1.1拦截器类381 25.1.2拦截器生命周期381 25.1.3拦截器和CDI38125.2使用拦截器381 25.2.1拦截方法调用382 25.2.2拦截生命周 ...

  7. 这是一个hibernate 联合主键的例子

    package com.bird.entity; import java.io.Serializable; import javax.persistence.Entity; import javax. ...

  8. [Leetcode][Python]53: Maximum Subarray

    # -*- coding: utf8 -*-'''__author__ = 'dabay.wang@gmail.com' 53: Maximum Subarrayhttps://leetcode.co ...

  9. 如何不让oracle使用linux的swap分区

    经常看到swap分区被使用,被缓存的内容本来是为了增加命中率,结果去不断换入换出,导致本地磁盘IO增加,影响访问速度.所以在内存充足的情况下,如果我们觉得不需要使用swap分区的时候,那就要想办法尽量 ...

  10. [poj 1144]Network[Tarjan求割点]

    题意: 求一个图的割点. 输入略特别: 先输入图中点的总数, 接下来每一行首先给出一个点u, 之后给出一系列与这个点相连的点(个数不定). 行数也不定, 用0作为终止. 这样的输入还是要保证以数字读入 ...