绑定方式的Service使用

在实现绑定服务时,最重要的是定义onBind()回调方法返回的接口,有三种方式:

1. 继承Binder类

2. 使用Messenger

3. 使用AIDL

这里对1,2方式进行分析。

1.继承Binder类

如果你的服务是你的应用程序的私有服务,并且跟客户端运行在同一个进程中,那么就应该通过继承Binder类来创建你的接口,并且佛从onBind()方法中返回这个接口的一个实例。客户端接收这个Binder对象,并且能够使用这个对象直接访问Binder类中实现的或Service中的公共方法。

使用官方SDK文档中的例子:

ServiceShow.java

package com.luoye.servicelearn;

import java.util.Random;

import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder; public class ServiceShow extends Service{ private final IBinder binder = new LocalBinder();//创建Service的内部IBinder对象 public class LocalBinder extends Binder
{
ServiceShow getService() //实现Binder自己的方法,这里返回Service对象的引用
{
return ServiceShow.this;
}
} public int getRandomNumber()
{
return (new Random()).nextInt(100);
} @Override
public IBinder onBind(Intent intent) {
// TODO Auto-generated method stub
//返回一个IBinder对象,即Service的内部IBinder对象,作为ServiceConnection中onServiceConnected方法的IBinder传入
return binder;
} }

MainActivity.java

package com.luoye.servicelearn;

import com.luoye.servicelearn.ServiceShow.LocalBinder;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.view.Menu;
import android.view.View;
import android.widget.Toast; public class MainActivity extends Activity { ServiceShow localService;
boolean bound = false; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
} @Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
} public void buttonOnClickListener(View v)
{
switch(v.getId())
{
case R.id.start_service1:
Intent intent_start = new Intent(this, ServiceShow.class);
//绑定Service 第二个参数为ServiceConnection对象,由于bindService是异步的,
//bindService()方法立即返回并且不返回IBinder对象,所以需要在ServiceConnection的方法中去处理
//BIND_AUTO_CREATE表示if Service不存在,即onCreate()
bindService(intent_start, connection, Context.BIND_AUTO_CREATE);
break;
case R.id.stop_service1:
if(bound)
{
unbindService(connection);
bound = false;
}//解绑Service
break;
case R.id.show_random:
if(bound == true)
{
Toast.makeText(this, "number is :"+localService.getRandomNumber(), Toast.LENGTH_SHORT).show();
}
break;
}
} private ServiceConnection connection = new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
LocalBinder binder = (LocalBinder)service; //得到service的IBinder对象
localService = binder.getService();//得到Service的引用,后续即可使用Service的方法
bound = true;
} @Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
bound = false;
} }; }

2. 使用Messenger

如果需要服务跟远程进程通信,那么就可以使用Messenger对象来给服务提供接口。

以下是信使(Messenger)对象的使用概要:

1.  服务端实现的一个处理器(Handler接口),这个处理器针对每次来自客户端的调用接收一次回调;

2.  这个处理器被用于创建一个信使对象(Messager)(这个信使对象要引用这个处理器);

3.  信使对象创建一个创建一个服务端从onBind()方法中返回给客户端的IBinder对象;

4.  客户端使用这个IBinder对象来实例化这个信使对象(信使引用了服务端的处理器),客户端使用这个信使给服务端发送Message对象;

5.  服务端在它的处理器(Handler)的handleMessage()方法中依次接收每个Message对象

在这种方法中,没有给客户端提供服务端的方法调用,相反,客户端会给发送服务端消息(Message)对象,服务端会在它的处理器中接受这些消息对象。

代码:

ServiceShow.java

package com.luoye.servicelearn;

import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.widget.Toast; public class ServiceShow extends Service{ static final int MSG_CODE = 0x10;//message的消息类型定义 class IncomingHandler extends Handler //message处理器,应与信使绑定
{
public void handleMessage(Message msg)
{
switch(msg.what)
{
case MSG_CODE:
Toast.makeText(ServiceShow.this, "Hello world", Toast.LENGTH_SHORT).show();
break;
default:
super.handleMessage(msg);
}
}
}
//创建一个信使,该信使使用Handler对象作为msg的处理器
final Messenger messenger = new Messenger(new IncomingHandler()); @Override
public IBinder onBind(Intent intent) {
// TODO Auto-generated method stub
Toast.makeText(ServiceShow.this, "Binding", Toast.LENGTH_SHORT).show();
return messenger.getBinder();//返回对应信使的Binder,返回给client使用
} }

MainActivity.java

package com.luoye.servicelearn;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.view.Menu;
import android.view.View; public class MainActivity extends Activity { Messenger messenger;
boolean bound = false; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
} @Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
} public void buttonOnClickListener(View v)
{
switch(v.getId())
{
case R.id.start_service1:
Intent intent_start = new Intent(this, ServiceShow.class);
//绑定Service 第二个参数为ServiceConnection对象,由于bindService是异步的,
//bindService()方法立即返回并且不返回IBinder对象,所以需要在ServiceConnection的方法中去处理
//BIND_AUTO_CREATE表示if Service不存在,即onCreate()
bindService(intent_start, connection, Context.BIND_AUTO_CREATE);
if(bound)
{
//产生message对象
Message msg = Message.obtain(null, ServiceShow.MSG_CODE, 0, 0);
try
{
//使用client端的信使发送message,该消息会通过Binder对应的信使递交到Service端的
//信使,并由该端信使的消息处理器Handler进行处理
messenger.send(msg);
}
catch (Exception e)
{
e.printStackTrace();
}
}
break;
case R.id.stop_service1:
if(bound)
{
unbindService(connection);//解绑Service
bound = false;
}
break;
}
} private ServiceConnection connection = new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
messenger = new Messenger(service); //利用Service返回的信使的对应的IBinder来创建client端的信使对象
bound = true;
} @Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
messenger = null;
bound = false;
} }; }

[Android学习笔记5]四大应用组件之一:Service 下的更多相关文章

  1. [Android学习笔记4]四大应用组件之一:Service 上

    一.什么是Service 一个Service就是一个能够在后台执行长时操作的应用程序组件,并且不提供用户界面.一个应用程序组件能够启动一个Service,即使用户切换到另一个应用程序,这个Servic ...

  2. Android学习笔记(十七)——数据库操作(下)

    //此系列博文是<第一行Android代码>的学习笔记,如有错漏,欢迎指正! 这一次我们来试一试升级数据库,并进行数据库的CRUD操作,其中, C 代表添加(Create) ,R 代表查询 ...

  3. Android学习笔记(五一):服务Service(上)- IntentService

    转自 http://blog.csdn.net/flowingflying/article/details/7616333 对于需要长期运行,例如播放音乐.长期和服务器的连接,即使已不是屏幕当前的ac ...

  4. Android学习笔记:使用ViewPager组件实现图片切换

    在很多App中,尤其是第一次安装启动后,都会出现几个图片进行一些app的介绍和说明,图片可以随着滑动而切换. 我们这里利用 ViewPager组件来演示如何实现这一点. 1.创建一个app工程,默认创 ...

  5. Android学习笔记 ImageSwitcher图片切换组件的使用

    activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...

  6. Android学习笔记 Toast屏幕提示组件的使用方法

    activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...

  7. Android学习笔记(24):进度条组件ProgressBar及其子类

    ProgressBar作为进度条组件使用,它还派生了SeekBar(拖动条)和RatingBar(星级评分条). ProgressBar支持的XML属性: Attribute Name Related ...

  8. Android学习笔记 TextSwitcher文本切换组件的使用

    activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...

  9. 【转】 Pro Android学习笔记(七六):服务(1):local和remote

    文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件.转载须注明出处:http://blog.csdn.net/flowingflying/ Android提供服务,服务是运行在后台的 ...

随机推荐

  1. Linq to Sqlite连接

    本人还是挺喜欢用Sqlite,鼓捣半天终于连上了,赶紧记录一下 1.当然还是新建一个项目,还是winform, 2.Vs2012添加NoGet,点击工具--扩展和更新,搜索NoGet,安装. 3.管理 ...

  2. struts2页面输出错误信息

    <package name="user" namespace="/user" extends="struts-default"> ...

  3. XML方式实现Spring声明式事务管理

    1.首先编写一个实体类 public class Dept { private int deptId; private String deptName; public int getDeptId() ...

  4. POJ 3630 Phone List(trie树的简单应用)

    题目链接:http://poj.org/problem?id=3630 题意:给你多个字符串,如果其中任意两个字符串满足一个是另一个的前缀,那么输出NO,否则输出YES 思路:简单的trie树应用,插 ...

  5. Android 开发笔记“关闭默认键盘”

    1.打开AndroidManifest.xml文件 2.在对应的activity中增加配置信息 android:windowSoftInputMode="stateHidden"

  6. 读取IOS的相应路径

    //    IOS相应路径 NSString* bundlePath = [[NSBundle mainBundle] bundlePath]; NSLog(@"bundlePath = % ...

  7. Linux软件间的依赖关系(转)

    Linux中的软件大部分是零碎的,其粒度比windows的小很多,软件之间的依赖关系很强烈,下面是自己的一些理解: 一.Linux中的软件依赖Linux中的软件依赖关系成一颗拓扑树结构,比如A直接或间 ...

  8. MySQL FEDERATED 存储引擎

    MySQL中针对不同的功能需求提供了不同的存储引擎.所谓的存储引擎也就是MySQL下特定接口的具体实现. FEDERATED是其中一个专门针对远程数据库的实现.一般情况下在本地数据库中建表会在数据库目 ...

  9. 由世纪互联运营的 Windows Azure 现已在中国正式发布

     我们非常高兴地公开发布由世纪互联运营的 Windows Azure,这标志着我们成为第一家在中国国内正式提供公共云平台技术的跨国公司.这一伟大成就的实现,得益于 Microsoft 与世纪互联的 ...

  10. nginx自定义模块记录上游服务器特定响应头

    功能,服务器通过扩展自定义命令,记录上游的服务器返回的特定响应头内容,记录到本地文件中 代码如下: /* * Copyright (C) Ciaos */ #include <ngx_confi ...