Osheep

时光不回头,当下最重要。

IntentService使用及原理

IntentService是Service的子类,由于Service里面不能做耗时的操作,所以Google提供了IntentService,在IntentService内维护了一个工作线程来处理耗时操作,当任务执行完后,IntentService会自动停止。另外,可以启动IntentService多次,而每一个耗时操作会以工作队列的方式在IntentService的onHandleIntent回调方法中执行,并且,每次只会执行一个工作线程,执行完第一个再执行第二个,以此类推。
看代码更好理解

public class MyService extends IntentService {
    //这里必须有一个空参数的构造实现父类的构造,否则会报异常
    //java.lang.InstantiationException: java.lang.Class<***.MyService> has no zero argument constructor
    public MyService() {
        super("");
    }
    
    @Override
    public void onCreate() {
        System.out.println("onCreate");
        super.onCreate();
    }

    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        System.out.println("onStartCommand");
        return super.onStartCommand(intent, flags, startId);

    }

    @Override
    public void onStart(@Nullable Intent intent, int startId) {
        System.out.println("onStart");
        super.onStart(intent, startId);
    }

    @Override
    public void onDestroy() {
        System.out.println("onDestroy");
        super.onDestroy();
    }

    //这个是IntentService的核心方法,它是通过串行来处理任务的,也就是一个一个来处理
    @Override
    protected void onHandleIntent(@Nullable Intent intent) {
        System.out.println("工作线程是: "+Thread.currentThread().getName());
        String task = intent.getStringExtra("task");
        System.out.println("任务是 :"+task);
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

然后看看Activity里面怎么使用这个Service

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        Intent intent = new Intent(this,MyService.class);
        intent.putExtra("task","播放音乐");
        startService(intent);
        intent.putExtra("task","播放视频");
        startService(intent);
        intent.putExtra("task","播放图片");
        startService(intent);
    }
}

看下结果:

14:49:31.465 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onCreate
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStartCommand
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStart
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStartCommand
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStart
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStartCommand
14:49:31.467 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onStart
14:49:31.467 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 工作线程是:IntentService[]
14:49:31.467 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 任务是 :播放音乐
14:49:33.468 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 工作线程是: IntentService[]
14:49:33.468 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 任务是 :播放视频
14:49:35.472 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 工作线程是: IntentService[]
14:49:35.472 18974-19008/com.dgtech.sss.intentservicedemo I/System.out: 任务是 :播放图片
14:49:37.477 18974-18974/com.dgtech.sss.intentservicedemo I/System.out: onDestroy

从结果中可以看出我们startService()执行了三次, onCreate()方法只执行了一次,说明只有一个Service实例, onStartCommand()和onStart()也执行了三次,关键是onHandleIntent()也执行了三次,而且这三次是串行的,也就是执行完一个再执行下一个,当最后一个任务执行完, onDestroy()便自动执行了

下面我们创建两个intent来试试

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Intent intent = new Intent(this,MyService.class);
        intent.putExtra("task","播放音乐");
        startService(intent);

        Intent intent2 = new Intent(this,MyService.class);
        intent2.putExtra("task","播放音乐2");
        startService(intent2);
    }
}

结果:

14:50:49.568 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onCreate
14:50:49.569 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onStartCommand
14:50:49.569 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onStart
14:50:49.570 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onStartCommand
14:50:49.570 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onStart
14:50:49.570 20321-20354/com.dgtech.sss.intentservicedemo I/System.out: 工作线程是: IntentService[]
14:50:49.570 20321-20354/com.dgtech.sss.intentservicedemo I/System.out: 任务是 :播放音乐
14:50:51.571 20321-20354/com.dgtech.sss.intentservicedemo I/System.out: 工作线程是: IntentService[]
14:50:51.572 20321-20354/com.dgtech.sss.intentservicedemo I/System.out: 任务是 :播放音乐2
14:50:53.574 20321-20321/com.dgtech.sss.intentservicedemo I/System.out: onDestroy

再一次印证了我们的结论

下面看一下IntentService的源码,很简单
从onCreate()入手

  @Override
    public void onCreate() {

        super.onCreate();
        //HandlerThread就是一个带有handler的thread,这里就是创建了一个线程,注意这个线程的名字
        //上面的日志打印中"工作线程是: IntentService[]"
        //因为我们在自定义的MyService里面,构造方法写的是super(""),如果写的不是空串,
        //上面的日志打印就应该是"工作线程是: IntentService[我们定义的名称]"
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        thread.start();
        //获取这个线程的looper
        mServiceLooper = thread.getLooper();
        //创建了一个handler
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }
private final class ServiceHandler extends Handler {
        public ServiceHandler(Looper looper) {
            super(looper);
        }

        @Override
        public void handleMessage(Message msg) {
            //处理逻辑是我们在MyService里面重写的这个方法
            onHandleIntent((Intent)msg.obj);
            //从这里可以看出,我们执行完任务后,service就销毁了
            stopSelf(msg.arg1);
        }
    }

再来看看onStart()

@Override
    public void onStart(@Nullable Intent intent, int startId) {
        Message msg = mServiceHandler.obtainMessage();
        msg.arg1 = startId;
        msg.obj = intent;
        mServiceHandler.sendMessage(msg);
    }

如果你了解handler的原理,这个源码太简单了
如果不了解handler,建议看看我写的
http://www.jianshu.com/p/d4415033349d

点赞