9.5.2使用IntentService
服务中的代码都是默认运行在主线程当中的,如果直接在服务里去处理一些耗时的逻辑,就很容易出现ANR(Application Not Responding)的情况,这个时候需要用到多线程,在服务的每个具体的方法里开启一个子线程,然后再这里面去处理那些耗时的逻辑。
需求:在服务的具体的方法里开启一个子线程处理耗时逻辑,并让该服务在执行完毕后自动停止。
代码示例:
public classMyService extends Service{
@Override
public IBinder onBind (Intent intent){
return null;
}
@Override
public int onStartCommand(Intent intent, intflags, int startId){
new Thread(new Runnable(){
@Override
public void run(){
//处理具体耗时的逻辑
stopSelf(); //服务在执行完毕后自动停止
}
}).start();
return super.onStartCommand(intent, flags, startId);
}
}
上面的做法:程序员容易忘记开启线程,活着忘记调用stopSelf()方法。
为了可以更简单地创建一个异步的、会自动停止的服务,android专门提供了一个IntentService类(集开启线程和自动停止于一身)。
代码示例:
public class MyIntentService extends IntentService{
public MyIntentService() {
super("MyIntentService"); //这里首先提供一个无参构造函数,而且必须在其内部调用父类的有参构造函数
}
//该方法必须复写
@Override //在该方法中去处理具体的逻辑,这个方法已经是在子线程中运行了
protected voidonHandleIntent(Intent intent) {
Log.d("MyIntentService", "Thread id is " + Thread.currentThread().getId());
}
@Override //这方法不一定要复写,在这里只是为了验证该服务最终会自动停止
public void onDestroy() {
super.onDestroy();
Log.d("MyIntentService", "onDestroy executed");
}
}
启动服务:
case R.id.start_intent_service:
//打印主线程的id
Log.d("MainActivity", "Thread id is " + Thread.currentThread().getId());
Intent intentService = new Intent(this,MyIntentService.class);
startService(intentService);//启动服务
注意:还需要在manifest中进行注册。