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. PHP文件缓存类

    <?php /** * @desc 文件缓存 */ class Cache{ const C_FILE = '/Runtime/'; private $dir = ''; const EXT = ...

  2. Entity Framewor中的 Migration

    http://www.entityframeworktutorial.net/code-first/code-based-migration-in-code-first.aspx = Code bas ...

  3. python连接redis002

    例子001. 通过StrictRedis模式连接到redis.并调用get命令读取了一个string类型的值. #!/usr/bin/python #!coding:utf-8 import redi ...

  4. SQL Server 查看当前活动的锁

    第一步: 要查看活动中的锁,如果日前根本就没有活动中的锁怎么办,还好我会自己做一把. begin tran         select * from dbo.Nums         with(ta ...

  5. inno setup 跳过(Welcome)欢迎界面

    原文 http://zwkufo.blog.163.com/blog/static/25882512010816049549/ 在InnoSetup中,我们很容易用 function ShouldSk ...

  6. perl 调用按钮输出到文本框

    sub push_b4 { #$txt -> insert('end'); #select $txt; system("expect c:\\\\expect.txt >expe ...

  7. chrome console 调试xpath

    chrome console F12->$x(“//title”) [<title>Online Tools for Software Developers (Free)</t ...

  8. C#/winform 旅游管理信息系统

    工具:Visual Studio 2015,sql server2014 1.系统概述 该旅游管理信息系统可以为游客和公司业务管理员提供服务.游客可以对旅游路线,旅游班次,旅游团,保险,导游,交通工具 ...

  9. 杭电 HDU 4608 I-number

    http://acm.hdu.edu.cn/showproblem.php?pid=4608 听说这个题是比赛的签到题......无语..... 问题:给你一个数x,求比它大的数y. y的要求: 1. ...

  10. K-近邻算法python实现

    内容主要来源于机器学习实战这本书.加上自己的理解. 1.KNN算法的简单描写叙述 K近期邻(k-Nearest Neighbor.KNN)分类算法能够说是最简单的机器学习算法了. 它採用測量不同特征值 ...