Android 四大组件详解(二) Service
2016-01-21 11:01
453 查看
一开始学Android的时候就开始听说Service的相关内容,但是真正接触到Service并且使用它是在几个月前的一次IM(即时通讯)的实现上,IM的实现是基于Mqtt协议的,所以有一个Service专门来负责IM相关的操作,有空也会简单介绍一下Mqtt相关的知识。
Service是后台的概念,但它是运行在主线程的,这点很重要,千万别跟Thread搞混,也就是说如果你在Service中执行耗时操作的话是会带来ANR(应用无响应)错误的。Service跟Activity一样也有自己的生命周期,下面就让我们从以下几点来简单介绍一下Service的内容吧。
1. 如何使用Service
2. Service启动方式(bindService跟startService)
3. Service生命周期
4. IntentService简介
5.Service细节知识点
一、如何使用Service
启动一个Service之后Service就独立运行。
首先要自己生成一个Service类(这里用JaymeService为例)继承自Service,然后实现相关的方法(比如onCreate, onDestroy之类执行自己想要的操作)。
如果你想要启动Service时候Service跟启动者(Activity或者Application)之间保持联系。
那你就先要写一个自己的Binder类(这里以JaymeBinder为例),JaymeBinder必须继承自Binder类,然后在JaymeBinder中实现一个getService方法把Service本身返回,然后在JaymeService中生成一个JaymeBinder对象(这里叫为mBinder)。这样我们就有了一个mBinder,然后还要重写JaymeService中的onBind方法,把mBinder返回,Service中所要做的事情就是这些。(具体代码看下面!)
在实现了Service之后,还要通过正确的方式启动Service才能保持启动者跟Service之间的联系(这里以MainActivity跟JaymeService绑定为例)。
首先在MainActivity中要生成一个ServiceConnection对象(具体代码如下)
这样我们在MainActivity中就有了ServiceConnection对象,那就只差最后一步了,通过bindService方法绑定Service。
首先我们跟启动Activity类似,先生成一个Intent对象,然后调用bindService方法就可以实现Service跟MainActivity的绑定,由于此时MainActivity拥有了JaymeService的引用,也就可以操作JaymeService了,具体代码如下。
最后再给出完整的MainActivity代码(包含后面要介绍的内容)
对应的layout文件activity_main.xml
二、Service的启动方式
上面花了比较大的篇幅去介绍如何使用Service,但是只是会用,具体Service的两种启动方式还是没有进行解释,下面就来简单解释一下启动Service的两种方式(startService跟bindService)。
startService方式启动Service
我们先来介绍简单的启动Service方式--startService,这种方法只要生成一个Intent对象,然后通过MainActivity的startService方法就可以启动一个Service(具体代码如下)。
这种方式启动Service之后Service跟启动者(这里是MainActivity)没有什么关系,启动之后Service就会运行自己的生命周期,Service的销毁什么的都跟启动者没有关系了。
startService方式启动Service的关闭方法(stopService)
如果你想要关闭通过startService方式启动的Activity,你也可以通过生成一个Intent对象,然后调用Activity的stopService方法,具体方法如下。
bindService方式启动Service
刚才我们介绍了比较简单的启动Service的方法,下面就来简单介绍一下绑定Service的方法,也就是用bindService方式启动Service。
这种方式启动Service的话就可以在Service跟启动者之间保持一定的联系,启动者也可以操作Service中的相关行为,这样就可以让一些后台的工作放到Service中,然后通过他们之间的关系来进行通信(比如IM中维持心跳连接的操作)。
上面也说了,要以这种方式启动Service,首先要生成一个ServiceConnection对象实现它的回调方法(也就是在回调方法中可以拿到Service中的相关引用达到操作Service的效果),然后在调用bindService的时候把ServiceConnection对象作为参数传进去就ok了,具体代码参看上面就好。
bindService启动方式启动Service的关闭方法
如果你想要关闭通过bindService方式启动的Service,你可以通过调用MainActivity中的stopService方法来关闭Service。这里要注意的一点就是你在调用unbindService的时候必须保证Service是处于bind的状态下的,不然会抛出异常(这也就是Main Activity 中mIsBind的作用)。下面具体看下代码
三、Service的生命周期
在介绍了Service的简单使用方式跟启动方式之后,就可以来探索一下Service的生命周期方法了,Service 的生命周期跟Activity的有一点点类似,话不多说,先上图,一切简单明了。
从图我们可以看出两种不同的启动方式Service的生命周期是不一样的(具体的生命周期演示最后会给出一个demo)。
通过startService方式启动
第一次启动调用onCreate->onStartCommand,再次启动当Service处于启动状态时只调用onStartCommand,然后Service自己关闭或者启动者把它关闭就进入onDestroy了。
通过bindService方式启动
第一次启动bindService(Service没有启动时)会调用onCreate->onBind,当已经bind的时候再启动bindService不触发任何回调,启动unbindService的时候会先调用onUnbind,如果此时Service没有跟别的启动者绑定就会进入onDestroy。
需要注意的地方:
当先启动startService还没有调用stopService的时候,此时第一次调用bindService跟unBindService没有任何异常,但当第二次或者更多次调用bindService、unbindService的时候会不触发onBind跟onUnbind(不知道这算不算一个bug,如果您知道希望指出)
四、IntentService的简介
如果你一开始认真看了这篇文章你就知道Service是运行在主线程的,所以要执行耗时操作的话也会带来ANR错误。而IntentService就是一种可以在里面执行耗时操作而不会带来ANR的Service。
具体方法就是自己写一个IntentService(这里使用JaymeIntentService为例)继承自IntentService,然后重写它的onHandleIntent函数,就可以在里面执行耗时的操作。
接着在启动者中调用startService就可以让IntentService执行耗时的操作了
五、Service细节知识点
1. Service是运行在主线程的,所以不要在Service中执行耗时操作,不然会带来ANR错误,要执行耗时操作的话要在Service建立新的线程去执行
2. ServiceConnection中的onServiceConnected会在bindService的时候回调,但是onServiceDisconnected在unbindService的时候不一定会调用,而是在Service
被系统强杀或者Service崩溃了的时候会调用
总结:
我们在了解Service的时候首先要了解怎么使用Service,然后熟悉它的两种启动方式(startService跟bindService),熟悉它的生命周期,这样才能熟练地操作Service的相关行为。
补上LogUtils的源码,主要是为了发布版本控制Log输出的需要。
六、Service演示demo
需要源码的可以去我的资源页查找下载(Android Studio源代码)
* 上面的一切观点都是个人的理解,如果有不准确的地方还希望大家指出 *
Service是后台的概念,但它是运行在主线程的,这点很重要,千万别跟Thread搞混,也就是说如果你在Service中执行耗时操作的话是会带来ANR(应用无响应)错误的。Service跟Activity一样也有自己的生命周期,下面就让我们从以下几点来简单介绍一下Service的内容吧。
1. 如何使用Service
2. Service启动方式(bindService跟startService)
3. Service生命周期
4. IntentService简介
5.Service细节知识点
一、如何使用Service
启动一个Service之后Service就独立运行。
首先要自己生成一个Service类(这里用JaymeService为例)继承自Service,然后实现相关的方法(比如onCreate, onDestroy之类执行自己想要的操作)。
如果你想要启动Service时候Service跟启动者(Activity或者Application)之间保持联系。
那你就先要写一个自己的Binder类(这里以JaymeBinder为例),JaymeBinder必须继承自Binder类,然后在JaymeBinder中实现一个getService方法把Service本身返回,然后在JaymeService中生成一个JaymeBinder对象(这里叫为mBinder)。这样我们就有了一个mBinder,然后还要重写JaymeService中的onBind方法,把mBinder返回,Service中所要做的事情就是这些。(具体代码看下面!)
package com.scut.jayme.services; import android.app.Service; import android.content.Intent; import android.os.Binder; import android.os.IBinder; import com.scut.jayme.utils.LogUtils; /** * 一个用于掌握 Service生命周期 的Service * * Created by jayme on 15/12/5. */ public class JaymeService extends Service{ private JaymeBinder mBinder = new JaymeBinder(); @Override public IBinder onBind(Intent intent) { showMessage("onBind"); return mBinder; } @Override public void onCreate() { showMessage("onCreate"); super.onCreate(); } @Override public void onDestroy() { showMessage("onDestroy"); super.onDestroy(); } @Override public int onStartCommand(Intent intent, int flags, int startId) { showMessage("onStartCommand"); return super.onStartCommand(intent, flags, startId); } @Override public boolean onUnbind(Intent intent) { showMessage("onUnbind"); return super.onUnbind(intent); } @Override public void onRebind(Intent intent) { showMessage("onRebind"); super.onRebind(intent); } private void showMessage(String message){ LogUtils.i("jaymeService", message); } /** * 用于跟Activity建立关联的Binder */ public class JaymeBinder extends Binder { public JaymeService getService(){ return JaymeService.this; } } }
在实现了Service之后,还要通过正确的方式启动Service才能保持启动者跟Service之间的联系(这里以MainActivity跟JaymeService绑定为例)。
首先在MainActivity中要生成一个ServiceConnection对象(具体代码如下)
private ServiceConnection mServiceConnection = new ServiceConnection() { /** * 当Service跟Activity绑定的时候会回调该函数 * @param name 名称 * @param binder JaymeService中的mBinder对象 */ @Override public void onServiceConnected(ComponentName name, IBinder binder) { LogUtils.i("jaymeActivity", "onServiceConnected"); /** 这里就获取到了Service对象,然后就可以操作Service了 */ mJaymeService = ((JaymeService.JaymeBinder)binder).getService(); } @Override public void onServiceDisconnected(ComponentName name) { LogUtils.i("jaymeActivity", "onServiceDisconnected"); } };
这样我们在MainActivity中就有了ServiceConnection对象,那就只差最后一步了,通过bindService方法绑定Service。
首先我们跟启动Activity类似,先生成一个Intent对象,然后调用bindService方法就可以实现Service跟MainActivity的绑定,由于此时MainActivity拥有了JaymeService的引用,也就可以操作JaymeService了,具体代码如下。
/** * 绑定Service操作 */ private void bindService(){ Intent bindIntent = new Intent(mContext, JaymeService.class); bindService(bindIntent, mServiceConnection, BIND_AUTO_CREATE); mIsBind = true; }
最后再给出完整的MainActivity代码(包含后面要介绍的内容)
package com.scut.jayme.activitys;
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.View;
import android.widget.Button;
import android.widget.Toast;
import com.scut.jayme.R;
import com.scut.jayme.services.JaymeIntentService;
import com.scut.jayme.services.JaymeService;
import com.scut.jayme.utils.LogUtils;
public class MainActivity extends Activity implements View.OnClickListener{
private Context mContext;
private boolean mIsBind = false;
private JaymeService mJaymeService;
private Button mStartServiceButton;
private Button mStopServiceButton;
private Button mBindServiceButton;
private Button mUnbindServiceButton;
private Button mStartIntentServiceButton;
private ServiceConnection mServiceConnection = new ServiceConnection() { /** * 当Service跟Activity绑定的时候会回调该函数 * @param name 名称 * @param binder JaymeService中的mBinder对象 */ @Override public void onServiceConnected(ComponentName name, IBinder binder) { LogUtils.i("jaymeActivity", "onServiceConnected"); /** 这里就获取到了Service对象,然后就可以操作Service了 */ mJaymeService = ((JaymeService.JaymeBinder)binder).getService(); } @Override public void onServiceDisconnected(ComponentName name) { LogUtils.i("jaymeActivity", "onServiceDisconnected"); } };
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mContext = MainActivity.this;
findViews();
initListener();
}
/**
* 查找相关的控件
*/
private void findViews(){
mStartServiceButton = (Button)findViewById(R.id.main_btn_start_service);
mStopServiceButton = (Button)findViewById(R.id.main_btn_stop_service);
mBindServiceButton = (Button)findViewById(R.id.main_btn_bind_service);
mUnbindServiceButton = (Button)findViewById(R.id.main_btn_unbind_service);
mStartIntentServiceButton = (Button)findViewById(R.id.main_btn_start_intent_service);
}
/**
* 初始化监听器
*/
private void initListener(){
mStartServiceButton.setOnClickListener(this);
mStopServiceButton.setOnClickListener(this);
mBindServiceButton.setOnClickListener(this);
mUnbindServiceButton.setOnClickListener(this);
mStartIntentServiceButton.setOnClickListener(this);
}
/** * 绑定Service操作 */ private void bindService(){ Intent bindIntent = new Intent(mContext, JaymeService.class); bindService(bindIntent, mServiceConnection, BIND_AUTO_CREATE); mIsBind = true; }
/** * 解除绑定Service操作 */ private void unbindService(){ if(mIsBind) { unbindService(mServiceConnection); mIsBind = false; }else{ Toast.makeText(mContext, "Service 已经解除绑定", Toast.LENGTH_SHORT).show(); } }
/** * 启动Service操作 */ private void startService(){ Intent startIntent = new Intent(mContext, JaymeService.class); startService(startIntent); }
/**
* 停止Service操作
*/
private void stopService(){
Intent stopIntent = new Intent(mContext, JaymeService.class);
stopService(stopIntent);
}
/** * 启动 IntentService */ private void startIntentService(){ for(int i = 0; i < 5; i++) { Intent intent = new Intent(mContext, JaymeIntentService.class); intent.putExtra("time", System.currentTimeMillis()); startService(intent); } }
@Override
public void onClick(View v) {
String toastMessage = null;
switch (v.getId()){
case R.id.main_btn_start_service:
toastMessage = "start service";
startService();
break;
case R.id.main_btn_stop_service:
toastMessage = "stop service";
stopService();
break;
case R.id.main_btn_bind_service:
toastMessage = "bind service";
bindService();
break;
case R.id.main_btn_unbind_service:
toastMessage = "unbind service";
unbindService();
break;
case R.id.main_btn_start_intent_service:
toastMessage = "start IntentService";
startIntentService();
break;
default:
break;
}
if(null != toastMessage){
Toast.makeText(mContext, toastMessage, Toast.LENGTH_SHORT).show();
}
}
}
对应的layout文件activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity"> <TextView android:text="通过下面操作跟Log信息检查学习Service的生命周期" android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/textView" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Bind" android:id="@+id/main_btn_bind_service" android:layout_below="@+id/textView" android:layout_alignParentRight="true" android:layout_alignParentEnd="true" android:layout_marginTop="44dp" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Start" android:id="@+id/main_btn_start_service" android:layout_alignTop="@+id/main_btn_bind_service" android:layout_alignParentLeft="true" android:layout_alignParentStart="true" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Stop" android:id="@+id/main_btn_stop_service" android:layout_below="@+id/main_btn_start_service" android:layout_alignRight="@+id/main_btn_start_service" android:layout_alignEnd="@+id/main_btn_start_service" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="UNBIND" android:id="@+id/main_btn_unbind_service" android:layout_alignTop="@+id/main_btn_stop_service" android:layout_alignParentRight="true" android:layout_alignParentEnd="true" android:layout_alignLeft="@+id/main_btn_bind_service" android:layout_alignStart="@+id/main_btn_bind_service" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="尝试三次启动IntentService" android:id="@+id/main_btn_start_intent_service" android:layout_below="@+id/main_btn_stop_service" android:layout_alignParentLeft="true" android:layout_alignParentStart="true" android:layout_marginTop="129dp" /> </RelativeLayout>
二、Service的启动方式
上面花了比较大的篇幅去介绍如何使用Service,但是只是会用,具体Service的两种启动方式还是没有进行解释,下面就来简单解释一下启动Service的两种方式(startService跟bindService)。
startService方式启动Service
我们先来介绍简单的启动Service方式--startService,这种方法只要生成一个Intent对象,然后通过MainActivity的startService方法就可以启动一个Service(具体代码如下)。
/** * 启动Service操作 */ private void startService(){ Intent startIntent = new Intent(mContext, JaymeService.class); startService(startIntent); }
这种方式启动Service之后Service跟启动者(这里是MainActivity)没有什么关系,启动之后Service就会运行自己的生命周期,Service的销毁什么的都跟启动者没有关系了。
startService方式启动Service的关闭方法(stopService)
如果你想要关闭通过startService方式启动的Activity,你也可以通过生成一个Intent对象,然后调用Activity的stopService方法,具体方法如下。
/** * 停止Service操作 */ private void stopService(){ Intent stopIntent = new Intent(mContext, JaymeService.class); stopService(stopIntent); }
bindService方式启动Service
刚才我们介绍了比较简单的启动Service的方法,下面就来简单介绍一下绑定Service的方法,也就是用bindService方式启动Service。
这种方式启动Service的话就可以在Service跟启动者之间保持一定的联系,启动者也可以操作Service中的相关行为,这样就可以让一些后台的工作放到Service中,然后通过他们之间的关系来进行通信(比如IM中维持心跳连接的操作)。
上面也说了,要以这种方式启动Service,首先要生成一个ServiceConnection对象实现它的回调方法(也就是在回调方法中可以拿到Service中的相关引用达到操作Service的效果),然后在调用bindService的时候把ServiceConnection对象作为参数传进去就ok了,具体代码参看上面就好。
bindService启动方式启动Service的关闭方法
如果你想要关闭通过bindService方式启动的Service,你可以通过调用MainActivity中的stopService方法来关闭Service。这里要注意的一点就是你在调用unbindService的时候必须保证Service是处于bind的状态下的,不然会抛出异常(这也就是Main Activity 中mIsBind的作用)。下面具体看下代码
/** * 解除绑定Service操作 */ private void unbindService(){ if(mIsBind) { unbindService(mServiceConnection); mIsBind = false; }else{ Toast.makeText(mContext, "Service 已经解除绑定", Toast.LENGTH_SHORT).show(); } }
三、Service的生命周期
在介绍了Service的简单使用方式跟启动方式之后,就可以来探索一下Service的生命周期方法了,Service 的生命周期跟Activity的有一点点类似,话不多说,先上图,一切简单明了。
从图我们可以看出两种不同的启动方式Service的生命周期是不一样的(具体的生命周期演示最后会给出一个demo)。
通过startService方式启动
第一次启动调用onCreate->onStartCommand,再次启动当Service处于启动状态时只调用onStartCommand,然后Service自己关闭或者启动者把它关闭就进入onDestroy了。
通过bindService方式启动
第一次启动bindService(Service没有启动时)会调用onCreate->onBind,当已经bind的时候再启动bindService不触发任何回调,启动unbindService的时候会先调用onUnbind,如果此时Service没有跟别的启动者绑定就会进入onDestroy。
需要注意的地方:
当先启动startService还没有调用stopService的时候,此时第一次调用bindService跟unBindService没有任何异常,但当第二次或者更多次调用bindService、unbindService的时候会不触发onBind跟onUnbind(不知道这算不算一个bug,如果您知道希望指出)
四、IntentService的简介
如果你一开始认真看了这篇文章你就知道Service是运行在主线程的,所以要执行耗时操作的话也会带来ANR错误。而IntentService就是一种可以在里面执行耗时操作而不会带来ANR的Service。
具体方法就是自己写一个IntentService(这里使用JaymeIntentService为例)继承自IntentService,然后重写它的onHandleIntent函数,就可以在里面执行耗时的操作。
package com.scut.jayme.services; import android.app.IntentService; import android.content.Intent; import com.scut.jayme.utils.LogUtils; /** * 用于测试 IntentService 的相关属性 * Created by jayme on 15/12/5. */ public class JaymeIntentService extends IntentService{ public JaymeIntentService() { super("JaymeIntentService"); } @Override protected void onHandleIntent(Intent intent) { try { Thread.sleep(500); } catch (InterruptedException e) { e.printStackTrace(); } long time = intent.getLongExtra("time", 0); LogUtils.i("JaymeIntentService", time + ""); } }
接着在启动者中调用startService就可以让IntentService执行耗时的操作了
/** * 启动 IntentService */ private void startIntentService(){ for(int i = 0; i < 5; i++) { Intent intent = new Intent(mContext, JaymeIntentService.class); intent.putExtra("time", System.currentTimeMillis()); startService(intent); } }
五、Service细节知识点
1. Service是运行在主线程的,所以不要在Service中执行耗时操作,不然会带来ANR错误,要执行耗时操作的话要在Service建立新的线程去执行
2. ServiceConnection中的onServiceConnected会在bindService的时候回调,但是onServiceDisconnected在unbindService的时候不一定会调用,而是在Service
被系统强杀或者Service崩溃了的时候会调用
总结:
我们在了解Service的时候首先要了解怎么使用Service,然后熟悉它的两种启动方式(startService跟bindService),熟悉它的生命周期,这样才能熟练地操作Service的相关行为。
补上LogUtils的源码,主要是为了发布版本控制Log输出的需要。
package com.scut.jayme.utils; import android.util.Log; /** * Created by jay on 15/12/5. */ public class LogUtils { private static final boolean SHOW_LOG = true; private static final String TAG = "liujie"; public static void i(String message){ if(SHOW_LOG) { Log.i(TAG, message); } } public static void i(String tag, String message){ if(SHOW_LOG){ Log.i(tag, message); } } }
六、Service演示demo
需要源码的可以去我的资源页查找下载(Android Studio源代码)
* 上面的一切观点都是个人的理解,如果有不准确的地方还希望大家指出 *
相关文章推荐
- Android textView字间距自定义LetterSpacingTextView
- Android之SharedPreferences详解
- Android官方课程总结笔记【多媒体之音频管理】
- 关于android6.0不能使用BLE
- Android抖动的输入框
- 在Android源码目录用javadoc生成API html文档
- Android自定义View——自定义样式
- Android 应用启动速度优化
- Android沉浸式状态栏SystemBarTint的使用方法
- Android Studio 关于so文件导入的若干方案
- Android 自定义对话框(控制大小、位置)
- Android Studio 新建项目的R文件丢失的解决方法
- Android开发套路收集整理与讨论
- android技巧:apk文件反编译以及签名打包
- Android高级Picasso
- 通过adb命令获取Android手机的IP地址
- [置顶]Android 项目开发 基于Web Service 服务的中英翻译软件(四)使用Sqlite “辅助” Web Service
- [置顶]Android 项目开发 基于Web Service 服务的中英翻译软件(三) Web Service服务 & Ksoap2 项目
- [置顶]Android 项目开发 基于Web Service 服务的中英翻译软件(一)功能介绍
- android adb root方法