KJFrameForAndroid框架学习——多线程管理

在Android开发中,由于不能再UI线程中做耗时操作,常常需要开启线程来做一些操作。但是这样一来就产生了一个问题,就是大量的线程并发执行,造成了线程维护的开销进而使得代码质量下降手机发烫又耗电。让我们来看一下KJFrameForAndroid框架是如何解决这个问题的。
KJFrameForAndroid框架项目地址:https://github.com/kymjs/KJFrameForAndroid

其实Android提供了一套专门用于异步处理的类,就是我们熟悉又模式的AsynTask类。
AsynTask类就是对Thread类的一个封装,并且加入了一些新的方法。那么我们先来看一下它对于Thread请求的一系列管理与封装。
在jdk中有这样一个集合类叫:BlockingQueue<>它是一个 Queue<E>的子类,支持两个附加操作的 Queue,这两个操作是:获取元素时等待队列变为非空,以及存储元素时等待空间变得可用。
BlockingQueue 方法以四种形式出现,对于不能立即满足但可能在将来某一时刻可以满足的操作,这四种形式的处理方式不同:第一种是抛出一个异常,第二种是返回一个特殊值(null 或 false,具体取决于操作),第三种是在操作可以成功前,无限期地阻塞当前线程,第四种是在放弃前只在给定的最大时间限制内阻塞。


我们来看一下在jdk中关于这个类的介绍:

BlockingQueue 不接受 null 元素。试图 add、put 或 offer 一个 null 元素时,某些实现会抛出 NullPointerException。null 被用作指示 poll 操作失败的警戒值。

好,为什么要讲这个集合类呢?因为这与我们接下来要讲的线程池维护有着莫大的关系。

?
1
2
     // 静态阻塞式队列,用来存放待执行的任务,初始容量:8个
     private  static  final  BlockingQueue<Runnable> mPoolWorkQueue =  new  LinkedBlockingQueue<Runnable>( 8 );

这里是KJFrameForAndroid开发框架中对于线程队列的维护对象,通过这个线程队列将所有线程放置在一个静态阻塞式队列中保存起来。
然而,仅仅是保存起来还不够,因为线程需要执行,我们必须要让线程进入到CPU工作间才行,这时KJFrameForAndroid使用了jdk中的另一个类ThreadPoolExecutor来管理并使用并发启动这些线程
来看一下在KJFrameForAndroid中对于这个类对象的创建

?
1
2
3
4
5
6
7
     /**
      * 并发线程池任务执行器,可以用来并行执行任务<br>
      * 与mSerialExecutor(串行)相对应
      */
     public  static  final  Executor mThreadPoolExecutor =  new  ThreadPoolExecutor(
             CORE_POOL_SIZE, MAXIMUM_POOL_SIZE, KEEP_ALIVE,
             TimeUnit.SECONDS, mPoolWorkQueue, mThreadFactory);

通过注释,我们还可以看到KJFrameForAndroid框架其实还有一种并行执行线程的方式,这里暂时不讲,我们看看这个类的构造方法在jdk中的解释:

那么至此我们有了并发任务执行器和线程池队列,最后就只需要将线程执行起来就行了。ThreadPoolExecutor类是一个实现了执行器Executor接口的类,那么它也就有一个execute方法去执行线程,我们所要做的就是调用这个类来执行它就可以了。

接下来我们来看一下如何让线程执行完成后又回到UI线程去做界面处理的。

还是看源码,以下是摘自KJFrameForAndroid开源框架中的一段代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
/*********************** start 一个完整的执行周期 ***************************/
     /**
      * 在doInBackground之前调用,用来做初始化工作 所在线程:UI线程
      */
     protected  void  onPreExecute() {}
     /**
      * 这个方法是我们必须要重写的,用来做后台计算 所在线程:后台线程
      */
     protected  abstract  Result doInBackground(Params... params);
     /**
      * 打印后台计算进度,onProgressUpdate会被调用<br>
      * 使用内部handle发送一个进度消息,让onProgressUpdate被调用
      */
     protected  final  void  publishProgress(Progress... values) {
         if  (!isCancelled()) {
             mHandler.obtainMessage(MESSAGE_POST_PROGRESS,
                     new  KJTaskResult<Progress>( this , values))
                     .sendToTarget();
         }
     }
     /**
      * 在publishProgress之后调用,用来更新计算进度 所在线程:UI线程
      */
     protected  void  onProgressUpdate(Progress... values) {}
     /**
      * 任务结束的时候会进行判断:如果任务没有被取消,则调用onPostExecute;否则调用onCancelled
      */
     private  void  finish(Result result) {
         if  (isCancelled()) {
             onCancelled(result);
         else  {
             onPostExecute(result);
         }
         mStatus = Status.FINISHED;
     }
     /**
      * 在doInBackground之后调用,用来接受后台计算结果更新UI 所在线程:UI线程
      */
     protected  void  onPostExecute(Result result) {}
     /**
      * 所在线程:UI线程<br>
      * doInBackground执行结束并且{@link #cancel(boolean)} 被调用。<br>
      * 如果本函数被调用则表示任务已被取消,这个时候onPostExecute不会再被调用。
      */
     protected  void  onCancelled(Result result) {}
     /*********************** end 一个完整的执行周期 ***************************/

在这里我们看到了三个非常熟悉的函数:doInBackground、onPostExecute、onProgressUpdate
没错,这三个函数正式SyncTask中被我们使用的三个方法(如果你还不知道,请搜索Android开发中SyncTask用法)接着我们再看看他们是如何被调用的:
这里声明了一个线程任务被执行的方法,实际上也就是doInBackground、被调用的方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/**
      * 必须在UI线程调用此方法<br>
      * 通过这个方法我们可以自定义KJTaskExecutor的执行方式,串行or并行,甚至可以采用自己的Executor 为了实现并行,
      * asyncTask.executeOnExecutor(KJTaskExecutor.mThreadPoolExecutor, params);
      */
     public  final  KJTaskExecutor<Params, Progress, Result> executeOnExecutor(
             Executor exec, Params... params) {
         if  (mStatus != Status.PENDING) {
             switch  (mStatus) {
             case  RUNNING:
                 throw  new  IllegalStateException(
                         "Cannot execute task:"
                                 " the task is already running." );
             case  FINISHED:
                 throw  new  IllegalStateException(
                         "Cannot execute task:"
                                 " the task has already been executed "
                                 "(a task can be executed only once)" );
             default :
                 break ;
             }
         }
         mStatus = Status.RUNNING;
         onPreExecute();
         mWorker.mParams = params;
         exec.execute(mFuture); // 原理{@link #execute(Runnable runnable)}
         // 接着会有#onProgressUpdate被调用,最后是#onPostExecute
         return  this ;
     }

而onProgressUpdate实际上是在这个handle中被调用了。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
      * KJTaskExecutor内部Handler,用来发送后台计算进度更新消息和计算完成消息
      */
     private  static  class  InternalHandler  extends  Handler {
         @Override
         @SuppressWarnings ({  "unchecked" "rawtypes"  })
         public  void  handleMessage(Message msg) {
             KJTaskResult result = (KJTaskResult) msg.obj;
             switch  (msg.what) {
             case  MESSAGE_POST_RESULT:
                 result.mTask.finish(result.mData[ 0 ]);
                 break ;
             case  MESSAGE_POST_PROGRESS:
                 result.mTask.onProgressUpdate(result.mData);
                 break ;
             }
         }
     }
有关本类的完整代码可以查看 https://github.com/kymjs/KJFrameForAndroid/blob/master/KJLibrary/src/org/kymjs/aframe/core/KJTaskExecutor.java

版权声明:本文为博主原创文章,未经博主允许不得转载。

转载于:https://my.oschina.net/kjframe/blog/484886

===========================各模块介绍====================== Plugin模块 你想像过不安装一个apk而直接被应用调用吗?你考虑过将应用的某一个功能单独出来像文件一样随时读取吗?插件化模块可以轻松帮你实现apk动态加载,让你的应用调用未安装的apk。 UILibrary模块包括两部分: Widget控件部分 主要封装了常用的UI控件,为了不让项目jar包过大,我们只引入了开发中一定会用到的控件,例如:可上下拉的KJListView、可上下拉的KJScrollView、可以双指缩放双击缩放双指旋转的ScaleImageView、等等......更多内容请自行查看项目文件中.widget包下的内容 Topology拓扑部分 包含一个使用IOC设计思想的控件初始化方式:可通过注解的方式进行UI绑定,与设置监听,在Activity和Fragment中均可以通过一行代码绑定控件并实现点击监听;还包含了在目前应用开发中常见的布局界面,如侧滑效果,高效的底部TAB导航,3D效果的切换。同时UILibrary为开发者定义了完善的BaseActivity和BaseFragment,开发者只需手动继承就可以获得Topology部分的全部功能。 BitmapLibrary模块 一行代码实现网络图片加载,任何View(ImageView设置src,普通View设置bg)加载图片的时候都无需考虑图片加载过程中出现的oom和android容器快速滑动时候出现的图片错位等现象,同时无需考虑图片加载过程中出现的OOM。默认使用内存lru算法+磁盘lru算法缓存图片,同时节省内存消耗默认采用控件的大小作为图片的大小加载图片。 HttpLibrary模块 可以一行代码实现Http请求、一行代码实现文件或图片的上传与下载。 kjh.download( url, params, callback); KJLibrary默认对所有Http通信的数据做了缓存处理,缓存时间为5分钟。这么做的目的不仅是为了节省用户手机流量,同时是为了减少服务器压力 同时不同于afinal的下载BUG(请看这里:http://my.oschina.net/kymjs/blog/355944),KJFrameForAndroid原生支持文件断点下载。 DBLibrary模块 可以一行代码对数据库进行增删改查等操作:kjdb.update(); kjdb.add(); ....... 包含了android中的orm框架,使用了线程池对sqlite进行操作,一行代码就可以进行增删改查。支持一对多,多对一等查询 帮助文档:http://kjframe.github.io/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值