美文网首页
Android独有的多线程

Android独有的多线程

作者: EighteenGray | 来源:发表于2017-05-01 16:05 被阅读48次

    一提多线程,脑子里立马浮现出各种关于线程池、线程同步、线程状态的词汇,恨不能把线程控制的像控件一样手拿把攥,俨然已将Android系统提供的线程机制抛诸脑后,今天小编就来八一八这些Android独有的多线程。

    一、AsyncTask

    Sync是同步,Async就是异步了。先看下异步任务的使用方法,再分析其源码实现。
    1.使用方法
    AsyncTask<Params, Progress, Result>
    第一个为入参,第二个为进度类型,第三个为结果类型。
    四个回调方法,
    onPreExecute(),主线程中执行准备工作。
    doInBackground(Params... params),线程池执行异步任务,可以调用publishProgress更新任务进度,它会调用onProgressUpdate方法。返回Result给onPostExecute()。
    onProgressUpdate(Progress... values),主线程执行,更新进度。
    onPostExecute(Result result),主线程执行,执行结果的返回值。
    onCancelled(),取消执行。

    2.源码解析

    1. execute()-->executeOnExecutor()-->在3.0之后,第一个为串行,第二个为并行
    2. onPreExecute()-->serialExecutor.execute(mFuture)用于任务的排队-->有一个ArrayDeque<Runnable>任务列表,新来的Runnable会被插入到队列尾部-->
    3. 然后调用scheduleNext() –>如果没有正在活动的任务,就会取出一个任务执行THREAD_POOL_EXECUTOR.execute(mActive),用于执行任务-->
    4. 执行WorkerRunnable的call()-->postResult(doInBackground(params))-->
    5. message.sendToTarget()-->在Handler中handleMessage,如果是PROGRESS,则回调onProgressUpdate;如果是RESULT,则finish()-->onCancelled/onPostExecute(result)。

    二、HandlerThread

    普通Thread用于执行耗时任务。
    HandlerThread是使用Handler的Thread,在run()中通过Looper.prepare(); Looper.loop(); 创建消息队列并开启循环。使用HandlerThread.getLooper()创建Handler,通过Handler发送消息/Runnable,Runnable被加入到队列中,然后Looper取出Runnable执行。这里主要是通过Handler切换到了子线程,也就是HandlerThread的线程,所以可以执行耗时任务。在这个Looper里面不是rr执行,而是顺序串行执行。
    HandlerThread handlerThread = new HandlerThread();
    handlerThread.start();
    Handler handler = new Handler(handlerThread.getLooper());
    handler.sendMessage(new Runnable());
    handlerThread.quit();

    三、IntentService

    IntentService是能执行耗时任务的Service,它具有Service的高优先级,可以执行一些高优先级的后台任务。
    IntentService就是如上面,在onCreate中生成了HandlerThread和Handler。
    在onStartCommand中,调用了onStart,Handler发送消息到消息队列,Handler在handlerMessage中回调onHandleIntent(Intent),在onHandleIntent已经是子线程,可以执行耗时任务,然后stopSelf()。通常的Handler在处理消息时是在ui线程执行,这里因为用的是HandlerThread的Looper生成的Handler,所以切换到了HandlerThread这个子线程中去执行耗时任务。
    每执行一个任务就启动一次onStartCommand。
    public class MyIntentService extends IntentService
    {
    private final static String TAG = "main";
    private String url_path="http://ww2.sinaimg.cn/bmiddle/9dc6852bjw1e8gk397jt9j20c8085dg6.jpg";
    public MyIntentService()
    {
    super("IntentSer");
    }
    @Override
    public void onCreate()
    {
    super.onCreate();
    }

     @Override
     public int onStartCommand(Intent intent, int flags, int startId) 
    

    {
    return super.onStartCommand(intent, flags, startId);
    }

    @Override
     public void onDestroy() 
    

    {
    super.onDestroy();
    }

     @Override
     protected void onHandleIntent(Intent intent) 
    

    {
    try {
    // 在设备应用目录下创建一个文件
    File file=new File(this.getFilesDir(), "weibo.jpg");
    FileOutputStream fos=new FileOutputStream(file);
    // 获取网络图片的输入流
    InputStream inputStream = new URL(url_path).openStream();
    // 把网络图片输入流写入文件的输出流中
    byte[] date=new byte[1024];
    int len=-1;
    while((len=inputStream.read(date))!=-1)
    {
    fos.write(date, 0, len);
    }

             fos.close();
             inputStream.close();    
             Log.i(TAG, "The file download is complete");
         } catch (MalformedURLException e) 
    

    {
    e.printStackTrace();
    } catch (IOException e)
    {
    e.printStackTrace();
    }
    }
    }

    四、FAQ

    最后附两个常见的小问题出现的原因。
    1.在非UI线程中更新UI原因
    ViewRootImpl类中的checkThread()方法用于检测当前是否UI线程,如果不是就会引发CalledFromWrongThreadException。而ViewRootImpl是在onResume结束后才创建完成。如果在onResume执行完成前,比如onStart、onCreate中子线程执行完毕,并修改了UI,此时View没有绘制出来,相当于设置了UI的一些属性,最终也不会调用checkThread方法,便不会报错。如果在onResume之后,比如延时几秒钟,ViewRootImpl已经创建完成,就会引发checkThread方法,从而报错。

    2.在子线程正确更新UI
    Activity.runOnUiThread(Runnable)
    View.post(Runnable)
    View.postDelayed(Runnable,long)

    相关文章

      网友评论

          本文标题:Android独有的多线程

          本文链接:https://www.haomeiwen.com/subject/qszftxtx.html