Android开发拾穗程序员首页投稿(暂停使用,暂停投稿)

Android异步操作之IntentService

2016-05-08  本文已影响305人  BlainPeng

IntentService是什么?

IntentService是一种继承了Service并且是一个抽象类的特殊Service,所以使用它时必须创建它的子类才能使用IntentService

使用IntentService的优势和局限

1、与service相比,它有以下优势:

2、 与普通线程相比,它有以下优势:

3、 局限性

工作任务队列是顺序执行的,如果一个任务正在IntentService中执行时,此时如果再发送一个新的任务请求,这个新的任务会只能一直等待,直到前面一个任务执行结束才执行,这也导致了它的工作效率低

基本用法

test.gif

非常简单的Demo,通过IntentService去访问百度。执行完成后,自己主动destory

public class MyIntentService extends IntentService {

String resultData = null;


public MyIntentService() {
    super("MyIntentService");
}


@Override
protected void onHandleIntent(Intent intent) {
    String urlData = intent.getDataString();
    URL url = null;
    HttpURLConnection conn = null;
    try {
        url = new URL(urlData);
        conn = (HttpURLConnection) url.openConnection();
        conn.setRequestMethod("GET");
        InputStream inputStream = conn.getInputStream();
        InputStreamReader isr = new InputStreamReader(inputStream);
        BufferedReader br = new BufferedReader(isr);
        String inputLine = "";
        while ((inputLine = br.readLine()) != null) {
            resultData += inputLine + "\n";
        }
        // 官方推荐通过LocalBroadcastManager来发送广播进行UI更新
        Intent dataIntent = new Intent(AppConstants.BROADCAST_ACTION);
        dataIntent.putExtra(AppConstants.BROADCAST_RESULT, resultData);
        LocalBroadcastManager.getInstance(this).sendBroadcast(dataIntent);
    } catch (MalformedURLException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }

}

@Override
public void onDestroy() {
    super.onDestroy();
    Toast.makeText(this,"MyIntentService onDestroy",Toast.LENGTH_SHORT).show();
}

主界面代码

public class MainActivity extends AppCompatActivity {

    private TextView mResult;
    private MyReceiver myReceiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mResult = (TextView) findViewById(R.id.tv_result);
        myReceiver = new MyReceiver(mResult);
        IntentFilter filter = new IntentFilter(AppConstants.BROADCAST_ACTION);
        LocalBroadcastManager.getInstance(this).registerReceiver(myReceiver, filter);
    }
    
    public void startService(View view) {

        Intent service = new Intent(this, MyIntentService.class);
        service.setData(Uri.parse(AppConstants.URL));
        startService(service);
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        LocalBroadcastManager.getInstance(this).unregisterReceiver(myReceiver);
    }
}

源码分析

@Override
public void onCreate() {
    // TODO: It would be nice to have an option to hold a partial wakelock
    // during processing, and to have a static startService(Context, Intent)
    // method that would launch the service & hand off a wakelock.

    super.onCreate();
    HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
    thread.start();

    mServiceLooper = thread.getLooper();
    mServiceHandler = new ServiceHandler(mServiceLooper);
}

当IntentService被第一次启动时,它的onCreate方法会被调用,在方法内部创建了一个HandlerThread,然后使用它的Looper来构造一个Handler对象mServiceHandler,这样通过mServiceHandler发送的消息最终都会在HandlerThread中执行。

当IntentService被create后,紧接着它的onStartCommand方法就会被调用一次,

 @Override
public int onStartCommand(Intent intent, int flags, int startId) {
    onStart(intent, startId);
    return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}

继续深入onStart方法

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

很明显,IntentService仅仅是通过mServiceHandler发送了一条消息,这个消息会在HandlerThread中被处理。

我们再来看看ServiceHandler源码

    private final class ServiceHandler extends Handler {
    public ServiceHandler(Looper looper) {
        super(looper);
    }

    @Override
    public void handleMessage(Message msg) {
        onHandleIntent((Intent)msg.obj);
        stopSelf(msg.arg1);
    }
}

mServiceHandler收到消息后,会将intent对象传递给onHandleIntent方法处理。注意这个intent对象的内容和外界的startService(intent)中的intent的内容是完全一致的,通过这个intent对象即可解析出外界启动startService是所传的参数,通过这些参数就可以区分具体的后台任务。

protected abstract void onHandleIntent(Intent intent);

onHandleIntent是一个抽象方法,我们需要在子类中实现它。它的作用是从Intent参数中区分具体的任务并执行这些任务。如果目前只存在一个后台任务,那么onHandleIntent方法执行这个任务后,stopSelf(int startId)会直接停止该服务;如果目前存在多个后台任务,那么onHandleIntent会等到执行完最后一个任务时,stopSelf(int startId)会直接停止该服务。

由于每执行一个后台任务就必须启动一次IntentService,而IntentService内部则通过消息的方式向HandlerThread请求执行任务,Handler中的Looper是顺序处理消息的,这就意味着IntentService也是顺序执行后台任务的,当有多个后台任务同时存在,这些后台任务会按照外界发起的顺序排队执行

另外,当onHandleIntent执行完后,这里是采用stopSelf(int startId)来停止该服务,而不是直接采用stopSelf(),这是因为stopSelf()会立刻停止服务,而这个时候可能还有其他消息未处理;而stopSelf(int startId)则会等待所有的消息都处理完毕后才终止服务

上一篇 下一篇

猜你喜欢

热点阅读