android开发之service详解
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我们下回分解。
版权声明:本文为博主原创文章,未经博主允许不得转载。若有错误地方,还望批评指正,不胜感激。
android开发之service详解的更多相关文章
- Android开发之MdiaPlayer详解
Android开发之MdiaPlayer详解 MediaPlayer类可用于控制音频/视频文件或流的播放,我曾在<Android开发之基于Service的音乐播放器>一文中介绍过它的使用. ...
- Android开发之InstanceState详解
Android开发之InstanceState详解 本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...
- Android开发之InstanceState详解(转)---利用其保存Activity状态
Android开发之InstanceState详解 本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...
- Android开发之AlarmManager详解
AlarmManager实质是一个全局的定时器,是Android中常用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括Activity,Service,BroadcastReceiver ...
- Android开发之InstanceState详解(转)
本文来自:http://www.cnblogs.com/hanyonglu/archive/2012/03/28/2420515.html 本文介绍Android中关于Activity的两个神秘方法: ...
- android开发之shape详解
很多时候,使用shape能够实现的效果,你用一张图片也能够实现,但问题是一张图片无论你怎么压缩,它都不可能比一个xml文件小,因此,为了获得一个高性能的手机App,我们在开发中应该遵循这样一个原则:能 ...
- Android开发之sharedpreferences 详解
SharedPreferences简介: 做软件开发应该都知道,很多软件会有配置文件,里面存放这程序运行当中的各个属性值,由于其配置信息并不多,如果采用数据库来存放并不划算,因为数据库连接跟操作等 ...
- [转]ANDROID开发之SQLite详解
SQLite简介 Google为Andriod的较大的数据处理提供了SQLite,他在数据存储.管理.维护等各方面都相当出色,功能也非常的强大.SQLite具备下列特点: 1.轻量级 使用 SQLit ...
- Android开发之recycleView详解代码,看完包你熟练掌握recycleView的用法。转自网络经典文章
来源 http://jinyudong.com/2014/11/13/Introduce-RecyclerView-%E4%B8%80/ 编辑推荐:稀土掘金,这是一个针对技术开发者的一个应用,你可以在 ...
随机推荐
- java入门(与C++的不同之处)封装篇
初学java,总是想将它与之前的C++做比较,看了慕客网的java入门视频,一直觉得在面向对象方面,它和C++有太多相同的地方,结果今天学到了两点不同之处,现在将它记录下来: 1. java的访问修 ...
- 既然HTTP1.1协议里每个连接默认都是持久连接,那么为何当今所有报文都在使用Connetion:Keep-Alive
说白了,如果你发起时有,那么服务器支持,回应时也会有,不支持,也就没有了.所以一般客户端都会默认带着发,服务端返回不返回就是服务端的事了. 1. 支不支持长连接,关键在于服务端是否支持. 如果服务端不 ...
- OpenCV for c++Builder
整理日: 20154/6 Borland C++BuilderでOpenCVを使う 確認 Turbo C++ 2007/03 1. ダウンロード&インストール http://sourcefor ...
- 利用Anaconda安装python后,如何安装opencv-python
利用Anaconda安装python后,想要安装opencv-python,但发现利用opencv-python的官方教程,没法实现opencv的安装 还好看到了另外一篇博客的方法,试一下,果然凑效 ...
- xcode5时代如何设置Architectures和Valid Architectures
目前ios的指令集有以下几种: 1,armv6,支持的机器iPhone,iPhone2,iPhone3G及对应的iTouch 2,armv7,支持的机器iPhone4,iPhone4S 3,armv7 ...
- Student's Morning
sgu242:http://acm.sgu.ru/problem.php?contest=0&problem=242 题意:把n个人分到m组,每一组至少2个人,每个人只能属于一个组.一开始把题 ...
- java Clone()克隆
转自:http://www.blogjava.net/orangelizq/archive/2007/10/17/153573.html 现在Clone已经不是一个新鲜词语了,伴随着“多莉”的产生这个 ...
- C++必备知识
新进C++程序员应在一年内完成学习“basic”类别知识点,两年内完成学习“advance”类别知识点,三到四年内完成学习“expert”1.基础(Basic)(1)变量与基本类型:(2)typede ...
- c++ 名字粉碎(name mangling)
转自Ibm: Name mangling is the encoding of function and variable names into unique names so that linker ...
- Linux 关机命令 重启命令
Linux centos重启命令: 1.reboot2.shutdown -r now 立刻重启(root用户使用)3.shutdown -r 10 过10分钟自动重启(root用户使用)4.shut ...