service作为android的四大组件之一,其重要性可想而知,在开发中,我们经常把一些不需要与用户进行交互的工作放在service中来完成,service运行在后台,这样有些人可能会产生错觉,以为service是运行在新线程中,其实不然,service也运行在主线程中,因此不能在service中进行耗时操作,否则会报ANR异常,但是我们可以在service中新开线程来进行耗时操作,比如下载等等。

先来说说service的两种绑定方式,一种是通过Context.startService()来启动,另一种是绑定的方式,通过bindService方法来实现。

第一种启动方式:我们一般在onStart()方法中启动service(使用方法startService()或者bindService()),在onStop方法中停止service(使用方法stopService()或者unbindService())。


先来看看第一种:


public class MyService1 extends Service {

    @Override
public void onCreate() {
Log.i("lenve", "onCreate()");
sendMsg2Activity("服务创建");
}
//废弃的方法,不再使用
@Override
public void onStart(Intent intent, int startId) {
super.onStart(intent, startId);
Log.i("lenve", "onStart()");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i("lenve", "onStartCommand()");
return super.onStartCommand(intent, flags, startId);
} private void sendMsg2Activity(String string) {
Intent intent = new Intent("service2activity");
intent.putExtra("msg", string);
this.sendBroadcast(intent);
}
@Override
public void onDestroy() {
super.onDestroy();
Log.i("lenve", "onDestroy()");
}
@Override
public IBinder onBind(Intent intent) {
Log.i("lenve", "onBind()");
return null;
} }

在Activity中启动Service

intent = new Intent(this, MyService1.class);
startService(intent);

停止service

stopService(intent);

通过打印日志我们可以发现,第一次调用service时会触发onCreate()方法,后面的调用则不会再执行onCreate()方法,无论我们启动了多少次Service,它都只有一个,即停止service只需执行一次即可。


第二种,通过绑定的方式启动service


如果要细分的话这种启动方式又分为两种:

1.设计接口,通过Ibinder实现接口来对Activity暴露方法供其调用。

接口:

public interface MyServiceInterface {

    public int add(int a,int b);
}

MyService.java

public class MyService1 extends Service {

    private IBinder myBinder = new MyBinder();
public class MyBinder extends Binder implements MyServiceInterface{ @Override
public int add(int a, int b) {
return a+b;
} }
@Override
public IBinder onBind(Intent intent) {
Log.i("lenve", "onBind()");
return myBinder;
} @Override
public void onCreate() {
super.onCreate();
Log.i("lenve", "onCreate()");
myBinder = new MyBinder();
} @Override
public void onDestroy() {
super.onDestroy();
Log.i("lenve", "onDestroy()");
} @Override
public boolean onUnbind(Intent intent) {
Log.i("lenve", "onUnbind()");
return super.onUnbind(intent);
} @Override
public void onRebind(Intent intent) {
super.onRebind(intent);
Log.i("lenve", "onRebind()");
} }

在activity中启动service并调用add方法:

public class MainActivity extends Activity {

    private TextView tv1;
private ServiceConnection conn;
private Intent intent;
private MyServiceInterface msi; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv1 = (TextView) this.findViewById(R.id.textView1);
intent = new Intent(this, MyService1.class);
Log.i("lenve", "" + 1);
conn = new ServiceConnection() { // 连接中断时调用
@Override
public void onServiceDisconnected(ComponentName name) {
} // 连接成功时调用
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
msi = (MyServiceInterface) service;
Log.i("lenve", "连接成功!");
}
};
bindService(intent, conn, BIND_AUTO_CREATE);
} public void onClick(View v) {
switch (v.getId()) {
case R.id.button1:
tv1.setText(msi.add(4, 5) + "");
break; default:
break;
}
} @Override
protected void onStop() {
super.onStop();
unbindService(conn);
}
}

我们通过接口,可以在Activity中调用service中的方法,这样便实现了两者之间的通信。

2.在onBinder()中返回Binder,而在继承Binder的类中直接返回当前service对象,通过该service对象来调用service中公开的方法。

service代码

public class MyService2 extends Service {

    private IBinder myBinder = new MyBinder();

    public class MyBinder extends Binder {
public MyService2 getService() {
return MyService2.this;
}
} @Override
public IBinder onBind(Intent intent) {
Log.i("lenve", "onBind()");
return myBinder;
} @Override
public void onCreate() {
super.onCreate();
Log.i("lenve", "onCreate()");
myBinder = new MyBinder();
} @Override
public void onDestroy() {
super.onDestroy();
Log.i("lenve", "onDestroy()");
} @Override
public boolean onUnbind(Intent intent) {
Log.i("lenve", "onUnbind()");
return super.onUnbind(intent);
} @Override
public void onRebind(Intent intent) {
super.onRebind(intent);
Log.i("lenve", "onRebind()");
} public int minus(int a, int b) {
return a - b;
}
}

activity中绑定:

public class MainActivity extends Activity {

    private TextView tv1;
private ServiceConnection conn;
private Intent intent;
private MyService2 ms2;
private MyServiceInterface msi; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv1 = (TextView) this.findViewById(R.id.textView1);
// intent = new Intent(this, MyService1.class);
intent = new Intent(this, MyService2.class);
conn = new ServiceConnection() { // 连接中断时调用
@Override
public void onServiceDisconnected(ComponentName name) {
} // 连接成功时调用
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// msi = (MyServiceInterface) service;
//拿到当前service对象
MyBinder mb = (MyBinder) service;
ms2 = mb.getService();
Log.i("lenve", "连接成功!");
}
};
bindService(intent, conn, BIND_AUTO_CREATE);
// new Thread(new Runnable() {
//
// @Override
// public void run() {
// try {
// Thread.sleep(2000);
// tv1.setText(msi.add(4, 5)+"");
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// }
// }).start();
} public void onClick(View v) {
switch (v.getId()) {
case R.id.button1:
//直接调用service中的public方法
tv1.setText(ms2.minus(9, 6)+"");
// tv1.setText(msi.add(4, 5) + ""); break; default:
break;
}
} @Override
protected void onStop() {
super.onStop();
unbindService(conn);
}
}

个人觉得第二种方式更好,不仅省事,而且交互也比较方便。

其实,如果想实现service和activity之间的通信还可以使用broadcast。

更高深的aidl我们下回分解。

本示例代码下载http://pan.baidu.com/s/1sjupRZZ

版权声明:本文为博主原创文章,未经博主允许不得转载。若有错误地方,还望批评指正,不胜感激。

android开发之service详解的更多相关文章

  1. Android开发之MdiaPlayer详解

    Android开发之MdiaPlayer详解 MediaPlayer类可用于控制音频/视频文件或流的播放,我曾在<Android开发之基于Service的音乐播放器>一文中介绍过它的使用. ...

  2. Android开发之InstanceState详解

    Android开发之InstanceState详解   本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...

  3. Android开发之InstanceState详解(转)---利用其保存Activity状态

    Android开发之InstanceState详解   本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...

  4. Android开发之AlarmManager详解

    AlarmManager实质是一个全局的定时器,是Android中常用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括Activity,Service,BroadcastReceiver ...

  5. Android开发之InstanceState详解(转)

    本文来自:http://www.cnblogs.com/hanyonglu/archive/2012/03/28/2420515.html 本文介绍Android中关于Activity的两个神秘方法: ...

  6. android开发之shape详解

    很多时候,使用shape能够实现的效果,你用一张图片也能够实现,但问题是一张图片无论你怎么压缩,它都不可能比一个xml文件小,因此,为了获得一个高性能的手机App,我们在开发中应该遵循这样一个原则:能 ...

  7. Android开发之sharedpreferences 详解

    SharedPreferences简介:   做软件开发应该都知道,很多软件会有配置文件,里面存放这程序运行当中的各个属性值,由于其配置信息并不多,如果采用数据库来存放并不划算,因为数据库连接跟操作等 ...

  8. [转]ANDROID开发之SQLite详解

    SQLite简介 Google为Andriod的较大的数据处理提供了SQLite,他在数据存储.管理.维护等各方面都相当出色,功能也非常的强大.SQLite具备下列特点: 1.轻量级 使用 SQLit ...

  9. Android开发之recycleView详解代码,看完包你熟练掌握recycleView的用法。转自网络经典文章

    来源 http://jinyudong.com/2014/11/13/Introduce-RecyclerView-%E4%B8%80/ 编辑推荐:稀土掘金,这是一个针对技术开发者的一个应用,你可以在 ...

随机推荐

  1. java入门(与C++的不同之处)封装篇

    初学java,总是想将它与之前的C++做比较,看了慕客网的java入门视频,一直觉得在面向对象方面,它和C++有太多相同的地方,结果今天学到了两点不同之处,现在将它记录下来: 1.  java的访问修 ...

  2. 既然HTTP1.1协议里每个连接默认都是持久连接,那么为何当今所有报文都在使用Connetion:Keep-Alive

    说白了,如果你发起时有,那么服务器支持,回应时也会有,不支持,也就没有了.所以一般客户端都会默认带着发,服务端返回不返回就是服务端的事了. 1. 支不支持长连接,关键在于服务端是否支持. 如果服务端不 ...

  3. OpenCV for c++Builder

    整理日: 20154/6 Borland C++BuilderでOpenCVを使う 確認 Turbo C++ 2007/03 1. ダウンロード&インストール http://sourcefor ...

  4. 利用Anaconda安装python后,如何安装opencv-python

    利用Anaconda安装python后,想要安装opencv-python,但发现利用opencv-python的官方教程,没法实现opencv的安装 还好看到了另外一篇博客的方法,试一下,果然凑效 ...

  5. xcode5时代如何设置Architectures和Valid Architectures

    目前ios的指令集有以下几种: 1,armv6,支持的机器iPhone,iPhone2,iPhone3G及对应的iTouch 2,armv7,支持的机器iPhone4,iPhone4S 3,armv7 ...

  6. Student's Morning

    sgu242:http://acm.sgu.ru/problem.php?contest=0&problem=242 题意:把n个人分到m组,每一组至少2个人,每个人只能属于一个组.一开始把题 ...

  7. java Clone()克隆

    转自:http://www.blogjava.net/orangelizq/archive/2007/10/17/153573.html 现在Clone已经不是一个新鲜词语了,伴随着“多莉”的产生这个 ...

  8. C++必备知识

    新进C++程序员应在一年内完成学习“basic”类别知识点,两年内完成学习“advance”类别知识点,三到四年内完成学习“expert”1.基础(Basic)(1)变量与基本类型:(2)typede ...

  9. c++ 名字粉碎(name mangling)

    转自Ibm: Name mangling is the encoding of function and variable names into unique names so that linker ...

  10. Linux 关机命令 重启命令

    Linux centos重启命令: 1.reboot2.shutdown -r now 立刻重启(root用户使用)3.shutdown -r 10 过10分钟自动重启(root用户使用)4.shut ...