zoukankan      html  css  js  c++  java
  • 深入 Service IntentService

    在 Android 开发中,我们或许会碰到这么一种业务需求,一项任务分成几个子任务,子任务按顺序先后执行,子任务全部执行完后,这项任务才算成功。那么,利用几个子线程顺序执行是可以达到这个目的的,但是每个线程必须去手动控制,而且得在一个子线程执行完后,再开启另一个子线程。或者,全部放到一个线程中让其顺序执行。这样都可以做到,但是,如果这是一个后台任务,就得放到Service里面,由于Service和Activity是同级的,所以,要执行耗时任务,就得在Service里面开子线程来执行。那么,有没有一种简单的方法来处理这个过程呢,答案就是IntentService。

    什么是IntentService,首先看看官方的解释:
    IntentService is a base class forServices that handle asynchronous requests (expressed asIntents) on demand. Clients send requests throughstartService(Intent) calls; the service is started as needed, handles each Intent in turn using a worker thread, and stops itself when it runs out of work

    简单说,IntentService是继承于Service并处理异步请求的一个类,在IntentService内有一个工作线程来处理耗时操作,启动IntentService的方式和启动传统Service一样,同时,当任务执行完后,IntentService会自动停止,而不需要我们去手动控制。另外,可以启动IntentService多次,而每一个耗时操作会以工作队列的方式在IntentService的onHandleIntent回调方法中执行,并且,每次只会执行一个工作线程,执行完第一个再执行第二个,以此类推。

    还有一个说明是:
    All requests are handled on a single worker thread -- they may take as long as necessary (and will not block the application's main loop), but only one request will be processed at a time.
    大致意思是:所有请求都在一个单线程中,不会阻塞应用程序的主线程(UI Thread),同一时间只处理一个请求。

    那么,用IntentService有什么好处呢?首先,我们省去了在Service中手动开线程的麻烦,第二,当操作完成时,我们不用手动停止 Service,第三,it's so easy to use!

    因为IntentService本身是继承自Service,所以在使用的时候要先在AndroidManifest.xml中注册,否则报错:Unable to start service Intent not found
    IntentService有7个方法,其中最重要的是onHandleIntent(),在这里调用worker线程来处理工作,每次只处理一个intent,像上面描述的,如果有多个,它会顺序处理,直到最后一个处理完毕,然后关闭自己。一点都不用我们操心,多好。
    再介绍另一个很有意思的方法,setIntentRedelivery()。从字面理解是设置intent重投递。如果设置为true,onStartCommand(Intent, int, int)将会返回START_REDELIVER_INTENT,如果onHandleIntent(Intent)返回之前进程死掉了,那么进程将会重新启动,intent重新投递,如果有大量的intent投递了,那么只保证最近的intent会被重投递。这个机制也很好,大家可以尝试着用。

    ok,接下来让我们来看看如何使用,我写了一个Demo来模拟耗时操作,每点击一次按钮会创建一个服务,但每个服务运行完后,才会执行下一个服务。

      1 /**
      2 * Description:IntentService服务
      3 */
      4 public class IntentServiceDemo extends IntentService {
      5 
      6 private static final String TAG = "@@@IntentServiceDemo";
      7 
      8 /**
      9 * 这里必须有一个空的构造方法,否则Android不会注册
     10 */
     11 public IntentServiceDemo() {
     12 
     13 this("");
     14 }
     15 
     16 public IntentServiceDemo(String name) {
     17 // 必须实现父类的构造方法
     18 super(name);
     19 }
     20 
     21 @Override
     22 public IBinder onBind(Intent intent) {
     23 
     24 Log.e(TAG, "onBind");
     25 return super.onBind(intent);
     26 }
     27 
     28 @Override
     29 public void onCreate() {
     30 
     31 super.onCreate();
     32 Log.e(TAG, "onCreate");
     33 }
     34 
     35 @Override
     36 public void onStart(Intent intent, int startId) {
     37 super.onStart(intent, startId);
     38 Log.e(TAG, "onStart");
     39 }
     40 
     41 @Override
     42 public int onStartCommand(Intent intent, int flags, int startId) {
     43 Log.e(TAG, "onStartCommand");
     44 return super.onStartCommand(intent, flags, startId);
     45 }
     46 
     47 @Override
     48 public void setIntentRedelivery(boolean enabled) {
     49 
     50 super.setIntentRedelivery(enabled);
     51 Log.e(TAG, "setIntentRedelivery");
     52 }
     53 
     54 @Override
     55 protected void onHandleIntent(Intent intent) {
     56 
     57 String msgRecv = intent.getStringExtra("param");
     58 
     59 for (int i = 1; i <= 3; i++) {
     60 
     61 String resultTxt = msgRecv
     62 + " 第 "
     63 + i
     64 + " 次执行,时间是:"
     65 + DateFormat.format("MM/dd/yy hh:mm:ss",
     66 System.currentTimeMillis());
     67 
     68 Log.e(TAG, resultTxt);
     69 try {
     70 Thread.sleep(2000);
     71 } catch (InterruptedException e) {
     72 e.printStackTrace();
     73 }
     74 }
     75 }
     76 
     77 @Override
     78 public void onDestroy() {
     79 super.onDestroy();
     80 Log.e(TAG, "onDestroy");
     81 }
     82 }
     83 
     84 /**
     85 * Description:测试IntentService服务页面
     86 */
     87 public class IntentServiceActivity extends Activity implements OnClickListener {
     88 
     89 private int mNumber = 1;
     90 
     91 @Override
     92 protected void onCreate(Bundle savedInstanceState) {
     93 super.onCreate(savedInstanceState);
     94 setContentView(R.layout.activity_intentservice);
     95 
     96 findViewById(R.id.btnIentService).setOnClickListener(this);
     97 }
     98 
     99 @Override
    100 public void onClick(View v) {
    101 
    102 switch (v.getId()) {
    103 
    104 case R.id.btnIentService:
    105 
    106 // 可以启动多次,每启动一次,就会新建一个 work thread,但IntentService的实例始终只有一个
    107 Intent startServiceIntent = new Intent(IntentServiceActivity.this,IntentServiceDemo.class);
    108 Bundle bundle = new Bundle();
    109 bundle.putString("param", "oper" + mNumber);
    110 startServiceIntent.putExtras(bundle);
    111 startService(startServiceIntent);
    112 
    113 mNumber += 1;
    114 break;
    115 }
    116 }
    117 }

    【运行结果,上来猛点Button三次】
    onCreate
    onStartCommand
    onStart
    oper1 第 1 次执行,时间是:08/26/14 02:29:18
    onStartCommand
    onStart
    onStartCommand
    onStart
    oper1 第 2 次执行,时间是:08/26/14 02:29:20
    oper1 第 3 次执行,时间是:08/26/14 02:29:22
    oper2 第 1 次执行,时间是:08/26/14 02:29:24
    oper2 第 2 次执行,时间是:08/26/14 02:29:26
    oper2 第 3 次执行,时间是:08/26/14 02:29:28
    oper3 第 1 次执行,时间是:08/26/14 02:29:30
    oper3 第 2 次执行,时间是:08/26/14 02:29:32
    oper3 第 3 次执行,时间是:08/26/14 02:29:34
    onDestroy

    可以看出:当 IntentService 运行时,再启动 IntentService 不会执行 onCreate 方法,而先执行 onStartCommand 后执行 onStart 方法,并且先开启的服务运行结束后,才会运行后边的服务。当最后的服务运行完毕,就会执行 onDestroy 方法,来结束服务。

  • 相关阅读:
    ABP框架系列之三:(Entity Framework Integration-实体框架集成)
    ABP框架系列之二:(Entity Framework Core-实体核心框架)
    ABP框架系列之一:(Entity-实体)
    MUI框架开发HTML5手机APP(一)--搭建第一个手机APP(转)
    树莓派安装mono
    【C#】VS2015开发环境的安装和配置(转)
    ABP框架
    WebApi系列~HttpClient的性能隐患(转)
    django中navie时间和aware时间详解
    js扩展运算符(spread)是三个点(...)
  • 原文地址:https://www.cnblogs.com/zx-blog/p/11835632.html
Copyright © 2011-2022 走看看