IntentService的使用及分析

由于Service运行在主线程中,我们就不能在service中直接做耗时的操作,只能通过开启一个子线程来执行耗时任务,当任务执行完毕后,我们要手动通过stopSelf()来停止我们的服务。但往往会有忘记开启线程或忘记停止服务的情况,为了解决这个问题,Android提供了可以简单创建一个异步的,会自动停止的服务——IntentService。

在看IntentService前,我们来学习以下HandlerThread,我们知道,当我们的Android进程建立时,系统会自动调用looper.prepare来初始化一个looper,这个looper存在于主线程,我们在主线程创建一个Hnadler对象,便可以通过发送消息和处理消息的操作,这个handler对象时不能在子线程中使用的。那么如果我们想在子线程中使用handler怎么办,其实我们可以在子线程中调用looper.prepare来初始化一个looper,之后便可以在子线程中使用handler。更简单的方法,就是使用HandlerThread.
HandlerThread实际上就是一个自带looper的Thread。他的用法很简单。
下面时我测试时写的一个demo

public class MainActivity extends AppCompatActivity {
    HandlerThread thread;
    Handler Mainhandler;//主线程Handler
    Handler mHandler;//子线程Handler
    TextView textView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView=findViewById(R.id.mytext);
        thread=new HandlerThread("myhanlder");
        thread.start();
        mHandler=new Handler(thread.getLooper()){
            @Override
            public void handleMessage(Message msg) {
                try {
                    Thread.sleep(1000);
                    Message message=new Message();
                    message.what=msg.what;
                    Mainhandler.sendMessage(message);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            }
        };
        Mainhandler=new Handler(){
            @Override
            public void handleMessage(Message msg) {
                textView.setText(msg.what+"");
            }
        };
        for (int i=0;i<10;i++){
            Message message=new Message();
            message.what=i;
            mHandler.sendMessage(message);
        }

    }
}
我们新建一个mHandler,由于该handler对象是根据子线程中的looper创建的,因此该handler的handlemessage的方法实际上是在子线程中执行的。在这里我在主线程中通过mhandler的sendmessge的方法发送消息,之后在其handlermessage中我们获取msg后可以执行相应的耗时操作,之后又使用MainHandler切回主线程,刷新界面。
现在来看看HandlerThread的源码:
 public class HandlerThread extends Thread {
    int mPriority;
    int mTid = -1;
    Looper mLooper;

    public HandlerThread(String name) {
        super(name);
        mPriority = Process.THREAD_PRIORITY_DEFAULT;
    }

    protected void onLooperPrepared() {
    }

    @Override
    public void run() {
        mTid = Process.myTid();
        Looper.prepare();
        synchronized (this) {
            mLooper = Looper.myLooper();
            notifyAll();
        }
        Process.setThreadPriority(mPriority);
        onLooperPrepared();
        Looper.loop();
        mTid = -1;
    }

其run方法内部,初始化了一个looper,注意这里使用synchronized代码块来保证我们的mlooper的初始化。onLooperPrepared默认是个空方法,它执行在looper.loop方法,因此在自定义handlerthread时,我们在这个方法中可以对我们的handler进行创建。
我们创建子线程handler时,使用 mHandler=new Handler(thread.getLooper());
那我们再来看getlooper()方法。

public Looper getLooper() {
        if (!isAlive()) {
            return null;
        }

        // If the thread has been started, wait until the looper has been created.
        synchronized (this) {
            while (isAlive() && mLooper == null) {
                try {
                    wait();
                } catch (InterruptedException e) {
                }
            }
        }
        return mLooper;
    }

在这里,重点看 synchronized 中代码,如果线程已经开始运行但是mlooper为空,证明looper对象还没初始化完成,由于我们的handler对象是在UI线程中创建的,而mlooper是在子线程中创建的,因此我们必须保证mlooper对象初始化成功后在返回,当mlooper为空,调用wait来停止当前线程,直到上面run方法执行后,调用notifyall来重新唤起线程,此时looper对象已经初始化完成,可以正确返回。

介绍完HandlerThread,我们再来看IntentService,其实IntentService内部使用到了HandlerThread。直接来看源码:

public abstract class IntentService extends Service {
    private volatile Looper mServiceLooper;
    private volatile ServiceHandler mServiceHandler;
    private String mName;
    private boolean mRedelivery;

    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);
        }
    }


    public IntentService(String name) {
        super();
        mName = name;
    }


    public void setIntentRedelivery(boolean enabled) {
        mRedelivery = enabled;
    }

    @Override
    public void onCreate() {
                super.onCreate();
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        thread.start();

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

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


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

    @Override
    public void onDestroy() {
        mServiceLooper.quit();
    }


    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }


    protected abstract void onHandleIntent(Intent intent);
}

在onCreate方法中,我们新建了一个HandlerThread,然后创建一个ServiceHandler,传入HandlerThread中的looper,这样ServiceHandler的handlemessage方法将会运行在子线程中。我们来看ServiceHandler的handlemessage中执行了onHandleIntent的方法,该方法时抽象方法,需要用户去实现,我们的代码逻辑写在此方法中即可。执行完毕后,将调用stopself方法来停止服务。
注意:回调完成后回调用 stopSelf(msg.arg1),注意这个msg.arg1是个int值,相当于一个请求的唯一标识。每发送一个请求,会生成一个唯一的标识,然后将请求放入队列,当全部执行完成(最后一个请求也就相当于getLastStartId == startId),或者当前发送的标识是最近发出的那一个(getLastStartId == startId),则会销毁我们的Service.
如果传入的是-1则直接销毁。
那么,当任务完成销毁Service回调onDestory,可以看到在onDestroy中释放了我们的Looper:mServiceLooper.quit()。
好,这就是IntentService的分析。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容