本地/远程Service 和Activity 的交方式(转)
android SDK提供了Service,用于类似*nix守护进程或者windows的服务。
Service有两种类型:
- 本地服务(Local Service):用于应用程序内部
- 远程服务(Remote Sercie):用于android系统内部的应用程序之间
前者用于实现应用程序自己的一些耗时任务,比如查询升级信息,并不占用应用程序比如Activity所属线程,而是单开线程后台执行,这样用户体验比较好。
后者可被其他应用程序复用,比如天气预报服务,其他应用程序不需要再写这样的服务,调用已有的即可。
编写不需和Activity交互的本地服务示例
本地服务编写比较简单。首先,要创建一个Service类,该类继承android的Service类。这里写了一个计数服务的类,每秒钟为计数器加一。在服务类的内部,还创建了一个线程,用于实现后台执行上述业务逻辑。
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的一些限制和规定:
- android支持String和CharSequence
- 如果需要在aidl中使用其他aidl接口类型,需要import,即使是在相同包结构下;
- android允许传递实现Parcelable接口的类,需要import;
- android支持集合接口类型List和Map,但是有一些限制,元素必须是基本型或者上述三种情况,不需要import集合接口类,但是需要对元素涉及到的类型import;
- 非基本数据类型,也不是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文件,具体见:
然后,需要在服务的aidl文件中修改如下:
package
com.easymorse;
import
com.easymorse.CountBean;
interface
ICountService {
CountBean getCount();
}
本地/远程Service 和Activity 的交方式(转)的更多相关文章
- Service与Activity通信 回调方式***
要实现service与activity的高强度通信用什么方法? service与activity之前的通信方式有很多,回调接口方式.观察者模式.广播.还有handler等,方法有很多,但要高强度地通信 ...
- android开发之使用Messenger实现service与activity交互
service与activity交互的方式有多种,这里说说使用Messenger来实现两者之间的交互. Service程序 public class MessengerService extends ...
- android service 本地 远程 总结
android编写Service入门 android SDK提供了Service,用于类似*nix守护进程或者windows的服务. Service有两种类型: 本地服务(Local Service) ...
- Android Activity与远程Service的通信学习总结
当一个Service在androidManifest中被声明为 process=":remote", 或者是还有一个应用程序中的Service时,即为远程Service, 远程的意 ...
- Android Service与Activity之间通信的几种方式
在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务,所以在我们实际开发中,就会常常遇到Activity与Service之间的通信,我们一般在Activ ...
- android 远程Service以及AIDL的跨进程通信
在Android中,Service是运行在主线程中的,如果在Service中处理一些耗时的操作,就会导致程序出现ANR. 但如果将本地的Service转换成一个远程的Service,就不会出现这样的问 ...
- Service与Activity与交流AIDL
深圳旅游月.终于回来了,做了很多个月,这些天来的东西会慢慢总结出来的.今天,我们正在谈论的Service小东西:沟通. 固定通信的做法比较,基本上按照写模板可以实现. 1.Service与Activi ...
- 一个简单的demo学习Android远程Service(AIDL的使用)
这是milo很早之前写在论坛上的一个帖子,现在整理出来,milo也复习一下一般来说Android 的四大组件都是运行在同一个进程中的,但远程Service运行在不同的进程里.这进程间的通信是使用了An ...
- Android—Service与Activity的交互
service-Android的四大组件之一.人称"后台服务"指其本身的运行并不依赖于用户可视的UI界面 实际开发中我们经常需要service和activity之间可以相互传递数据 ...
随机推荐
- 恶补ASP.NET基础【1】枚举和结构
有时我们希望变量提取的是一个固定集合中的值,此时就可以用枚举类型, 例: enum OpenMode : byte { 新增=, 编辑=, 查看= } class Program { static v ...
- VS2012常用快捷建(必备)
1. 强迫智能感知:Ctrl+J:2.强迫智能感知显示参数信息:Ctrl-Shift-空格:3.格式化整个块:Ctrl+K+F4. 检查括号匹配(在左右括号间切换): Ctrl +]5. 选中从光标起 ...
- Nginx 变量漫谈(一)
Nginx 的配置文件使用的就是一门微型的编程语言,许多真实世界里的 Nginx 配置文件其实就是一个一个的小程序.当然,是不是“图灵完全的”暂且不论,至少据我观察,它在设计上受 Perl 和 Bou ...
- SQL 临时表或表变量替代游标
1.如果表没有自动增长的标识列(int) 使用临时表 SELECT IDENTITY(int) NewID ,.. INTO #tmp FROM YouTable 2.表有标识列 使用表变量 INSE ...
- Visual C++编程命名规则
一.程序风格: 1.严格采用阶梯层次组织程序代码: 各层次缩进的分格采用VC的缺省风格,即每层次缩进为4格,括号位于下一行.要求相匹配的大括号在同一列,对继行则要求再缩进4格.例如 ...
- retire
retire 本来抱着进队的决心迎战,结果在第一试就失利,是能力不足,还是命中注定我不是竞赛的料,一切关于OI的事随着GDOI2016的闭幕而消散-- 今后也许再也不搞OI了,或许会玩一下ACM,现在 ...
- IIS给网站地址配置成HTTPS的
2.增加网站绑定 3.如果https的URL访问不了,可能是443端口被占用 然后netstat -anono
- Node.js log3:create ejs engine and jade engine
The base condition is ensure that you have installed express. 1.create ejs engine Using windows d ...
- hdu 1181 变形课(dfs)
Problem Description 呃......变形课上Harry碰到了一点小麻烦,因为他并不像Hermione那样能够记住所有的咒语而随意的将一个棒球变成刺猬什么的,但是他发现了变形咒语的一个 ...
- 张冬:OpenPOWER CAPI为什么这么快?(二)
张冬:OpenPOWER CAPI为什么这么快?(二) PMC公司数据中心存储架构师张冬 有了CAPI的FPGA是怎么做的? 首先认识一下这个体系里的三个角色: AFU(Acceleration ...