美文网首页Android developer源码的解析Android
Android 深入理解Loader机制 让APP轻装上阵

Android 深入理解Loader机制 让APP轻装上阵

作者: Tamic | 来源:发表于2016-05-30 19:44 被阅读10264次

    Android开发者都经历过APP UI开发不当 会造成overDraw,导致APP UI渲染过慢,但是很多人却没听过overLoad,overLoad一般是由于开发者在主线程操作耗时操作,导致程序变慢 甚至出现的anr的现象,那么android早已为这种现象提供完美的解决方案,就是今天给大家说的Loader机制。

    一 Loader
    Android的装载器(loader)是从Android 3.0新引入的API , 主要完成单线程耗时数据异步装载功能,并在数据有更新自动通知UI刷新的作用。业内也叫加载器,装载机。
    Loader用途
    Loader一般用在Activity和fragment异步加载数据,无需重新启动一个线程来执行数据加载,异步加载可以用asyncTask, 但是loader自带数据结果监听机制,可以方便优雅的进行UI更新。

    作用和优点:

    提供异步加载数据功能;
    对数据源变化进行监听,实时更新数据;
    在Activity配置发生变化(如横竖屏切换)时不避免数据重复加载;
    适用于任何Activity和Fragment;

    加载耗时数据常用方式
    android开发者都知道不能再UI线程里去执行耗时操作,甚至在4.0里已经无法在主线程里去访问网络,那么一般加载耗时操作有以下办法。

    1 2B加载


    2b.jpg

    2 普通加载

    普通.jpg

    3 文艺加载

    使用Loader.jpg

    为何说1和2是不可取呢,我们从Loader源码看起,文章结束会知道答案。

    二 Loader实现

    Loader源码在android.content下面,可见它的份量有多重,loader机制包括LoaderManager,Loader,LoaderCallbacks三部分,
    LoaderManager 来管理我们的laoder实例,获取,初始化,重启一个loader,
    Loader 来执行我们的异步操作,有开始,完成,后台加载中等接口实现
    LoaderCallbacks 来执行我们的loader回调,主要是绑定分发Loader,完成加载,重置数据等。
    流程如下图:


    图片1.png

    1 LoaderManager
    LoaderManager是抽象类,负责管理一组Loader,主要定义执行Loader的一些抽象方法,类结构如下图:

    LoadManager.jpg

    从上图看以看出,Ta里面主要初始化loader,获取 重启,销毁一个loader,也包含一个内部成员变量LoaderCallback回调,主要方便我们在上层写回调实现累操作,但真正是由他的实现类LoaderManagerImpl去完成操作的,
    LoaderManagerImpl 记录着一组LoaderInfo信息,(ps:Activity也同等拥有activityInfo一样,只不过记录是Activityrecord来完成,其实谷歌很多源码都是相同的),并且持有LoaderManager.LoaderCallbacks, mLoader等成员,负责对Loader和LoaderCallbacks的对应回调,内部基于观察者模式实现,源码不在解读;

    2 Loader
    Loader是具体来操作任务的类,负责去调用不同渠道的数据接口,比如数据库,contentProvider, 文件等。

    loader.jpg

    从大致的UML图我可以了解loader持有一个内部观察者,和一些注册注销观者的内部方法,并且已经暴露出来的加载操作的状态步骤的方法,包括加载中,取消加载,强制加载,内容发生改变等,
    在平常的开发中,谷歌为我们提供了laoder的子类,AsyncTaskLoader,CursorLoader等子类, 源码不在介绍,现在说下他们的不同点。CursorLoader也是AsyncTaskLoader的子类,主要负责数据库查询的异步加载,AsyncTaskLoader可用来所有异步加载。

    2.1 AsyncTaskLoader
    AsyncTaskLoader继承了Loader, 除了拥有loader的功能,还有executePendingTask(), dispatchOnCancelled(),onLoadInBackground()等方法,最神奇的是他拥有AsyncTask的实例,并且实现Runnable,这是他能进行异步的原因所在。对AsyncTask不熟悉的请自我补脑,看如下代码,

    AsyncTaskLoader.jpg

    笔者看了源码,AsyncTaskLoader拥有AsyncTask,在自身实例化后开启一个线程,自我进行executePendingTask(),此方法里其实就在执行asyncTask的mTask.executeOnExecutor(mExecutor,(Void[])null);�来实现AsyncTaskLoader的自我监听机制,当然自身轮询和通信是离不开Handler的 因为整个android的通讯就是建立在Handler(底层binder)基础上,这里不再分析。

    2.2 CursorLoader
    CursorLoader是AsyncTaskLoader的子类,内部持有ForceLoadContentObserver变量,观察者来实现对数据源的数据更新,执行加载数据操作,最重要的当然是离不开查询操作,内部主要代码:

    CursorLoader

    三 怎么使用loader

    1 启动一个Loader
    Activity初始化在oncreate()初始化,一个Activity或Fragment中LoaderManager管理一个或多个Loader实例,每个Activity或Fragment只有一个LoaderManager,我们可以在Activity的onCreate()或Fragment的onActivityCreated()里初始化一个Loader。例如:
    getLoaderManager().initLoader(0, null, new DataLoaderCallback());

    可以看见上面的initLoader()方法有三个参数:

    第一个参数代表当前Loader的ID ,用来区分哪个loader;

    第二个参数代表提供给Loader构造函数的参数,Bundle对象类型 ,可选;

    第三个参数代表LoaderManager.LoaderCallbacks的回调实现 需要我自我实现。

    上面initLoader()方法的调用一个Loader被初始化和激活的状态,该方法的调运有如下两种结果:

    如果代表该Loader的ID已经存在,则后面创建的Loader将直接复用已经存在的;

    如果代表该Loader的ID不存在,initLoader()会触发LoaderManager.LoaderCallbacks回调的onCreateLoader()方法创建一个Loader;

    可以看见通过initLoader()方法可以将LoaderManager.LoaderCallbacks实例与Loader进行关联,且当Loader的状态变化时就被回调。所以说,如果调用者正处于其开始状态并且被请求的Loader已经存在,且已产生了数据,那么系统会立即调用onLoadFinished()(在initLoader()调用期间),所以你必须考虑到这种情况的发生。

    当然了,intiLoader()会返回一个创建的Loader,但是你不用获取它的引用,因为LoadeManager会自动管理该Loader的生命周期,你只用在它回调提供的生命周期方法中做自己数据逻辑的处理即可。

    2 实现LoaderManager.Callbacks回调

    LoaderManager.LoaderCallbacks是LoaderManager的回调交互接口。LoaderManager.LoaderCallbacks包含以下三个方法:

    onCreateLoader()
    实例化并返回一个新创建给指定ID的Loader对象;第一启动时调用

    onLoadFinished()
    load完成之后回调此方法;每次都调用

    onLoaderReset()
    当创建好的Loader被reset时调用此方法,会清空已绑定数据,此时CreatLoader会重新执行

    3 Loader使用实例

    1》 初始化loader

    getLoaderManager().initLoader(0, null, new DataLoaderCallback());

    2》实现callback接口,注册自我监听回调

    callback

    当然你也可以用来绑定谷歌提供的CursorLoader ,在Loader创建的时候被调用,这里使用一个ContentProvider获取数据,所以使用CursorLoader返回数据

    CursorLoader

    3》 继承Loader,构造自我的数据绑定,和数据适配

    Loader

    在这里我们模拟了构造一组数据,当然你也可以在loadInBackgruond去读文件,访问网络,查询数据库

    4 拓展
    1》 用来自动刷新ContentPorvider

    在我们使用CurSorLoader时大家都会考虑一种情况的处理—–当数据库发生变化时如何自动刷新当前UI,数据库在数据改变时通过ContentPorvider和ContentResolver发出通知,接着ContentProvider通知Cursor的观察者数据发生了变化,然后Cursor通知CursorLoader的观察者数据发生了变化,CursorLoader又通过ContentProvider加载新数据,完成后调用CursorAdapter的changeCursor()用新数据替换旧数据显示。

    这个过程具体的实现步骤如下:

    对获取的Cursor数据设置需要监听的URI(即,在ContentProvider的query()方法或者Loader的loadingBackground()方法中调用Cursor的setNotificationUri()方法);

    在ContentProvider的insert()、update()、delete()等方法中调用ContentResolver的notifyChange()方法;

    通过上面两步我们就能实现CurSorLoader的自动数据刷新功能了;可以发现,所谓的CurSorLoader自动刷新也是对文章开头说的观察者模式,所以不再过多说明。
    2》不使用ContentPorvider的自动刷新


    自动数据刷

    四Loaders相关源码流程

    通过上面我们的源码分析和分析前那副图可以总结如下结论:

    一次完整的数据加载流程为Activity调用LoaderManager的doStart()方法,LoaderManager调用Loader的startLoading()方法,然后Loader调运AsyncTaskLoader的doingBackground()方法进行耗时数据加载,紧接着AsyncTaskLoader回调LoaderManager的complete数据加载完成方法,接着又LoaderManager回调我们在Activity中实现的callback中的onLoadFinish()方法。

    Acivity和Fragment的生命周期主动管理了LoaderManager,每个Activity用一个ArrayMap的mAllLoaderManager来保存当前Activity及其附属Frament的唯一LoaderManager;在Activity配置发生变化时,Activity在destory前会保存mAllLoaderManager,当Activity再重新创建时,会在Activity的onAttcach()、onCreate()、performStart()方法中恢复mAllLoaderManager。

    LoaderManager给Activity提供了管理自己的一些方法;同时主动管理了对应的Loader,它把每一个Loader封装为LoadInfo对象,同时它负责主动调运管理Loader的startLoading()、stopLoading()、,forceLoad()等方法。

    由于整个Activity和Fragment主动管理了Loader,所以关于Loader的释放(譬如Cursor要要主动关闭游标的等,文件流要置空等)不需要我们人为处理,Loader会帮我们很好的处理的;同时特别注意,对于CursorLoader,当我们数据源发生变化时Loader框架会通过ContentObserver调用onContentChanged的forceLoad方法重新请求数据进行回调刷新。

    五 总结
    通过前面基础实例、源码分析、拓展你会发现Loader很强大,例如在普通展现某个android手机有多少应用程序,加载已安装app时候,其实loader就能排上用场。
    详细见谷歌对Loader介绍:
    https://developer.android.com/reference/android/content/AsyncTaskLoader.html

    PS:顺便说下AsyncTaskLoader与AsyncTask的区别,看完源码我们再回过头来总结性的说说他们二者区别,如下:

    两者区别

    最主要是加载数据,使用loader我们无须关注数据何时改变了,也无需关注activity的生命周期,做到数据不被重复多次加载情况,activty销毁数据自动释放的作用,做到一次加载多次使用的效果,我们可以依据需求,拿着loader变活灵通,这里的博大精深还需要你自己体会。
    案例:https://github.com/NeglectedByBoss/Loader
    更多文章请猛戳

    第一时间获取大佬技术文章和行业动态请关注微信公众号!

    开发者技术前线

    相关文章

      网友评论

      • 轲叶:讲解的非常不错,只是用的人确实少。从学习的角度,loader的机制是非常值得学习的。
        从使用的角度,如果只是为了异步操作,或是实时更新数据,感觉RxJava或DataBinding更加方便,而且装逼:smirk:
      • qweorqwrq:我理解的android系统,google官方做的东西更多偏向是技术展示给开发者看的。不是让开发者直接拿来用的。他提供了思路,你需要根据自己的需求和实际掌握程度来选择适合自己的技术。这点和ios很不同。ios很多东西都给你封装的好好的,你只需要知道怎么使用就好了。android很多东西做的都像是个半成品,需要自己去修改使用。
      • qweorqwrq:其实android系统有些api设计的很不合理。比如屏幕旋转,不同的系统版本旋转生命周期都有可能不一样。实际上很少有app会为了竖屏和横屏切换使用不同的布局。事情反而被搞复杂了。
      • qweorqwrq:和系统版本耦合严重的类不建议使用。尤其是不同的系统版本实现不一样的话,搞不好会搞死你。我好像没看到谁用过。
        MrTrying:@Madao_Jiang API版本问题,如果想用loader的话建议使用V4包下的loader,这应该不存在API版本问题了
        MrTrying:@a442509097 Thread+handler说是2B有点过了,有时候可能在主线程直接执行可能效率更高
        有_风:赞同, 感觉作者这篇文章不太客观, Thread + Handler被说成了很2B的写法, 既然如此为什么还是有很多人用, 就说明其适用的场景比较多, 比较好用.
      • b973150428c2:DataLoaderCallback中onCreateLoader方法中是不是要调用一下loader.forceLoad();才会去执行loader的loadInBackground()方法.我的程序是这样的,不知道大神的程序是不是这样,
      • xwp:ActivtyDestroy后,里面的Loder网络请求会继续吗?
        moonlandingplan:正常情况下,Loader会随着Activity或者Fragment一起销毁的。例外情况,如转屏导致Activity重建,那么Loader还是会被保留,重建后获取到的Loader就是页面销毁前的。
      • a树:为啥loader不是很常用呢
        Tamic: @a树 谷歌推荐用的 这估计和国人的开发习惯有关吧
      • 捡淑:马克
      • 市井小民whu:感觉这个loader的设计类似于presenter,不知道理解的对不对
        a树: @市井小民whu loader主要是加载耗时操作,而presenter主要是将逻辑与显示分离,二者有一小部分重叠,但侧重点不一样。
        Tamic:@市井小民whu 还是不一样的
      • 相互交流:LZ如果我在同一个页面,会有多个网络请求,,用同一个Loader会出现问题吗?,,或者说,用多个Loader,会有什么不好的地方吗??
        Tamic:@相互交流 没啥区别 就类似给onclick()一样 你分别注册就不用区分 ,用同一个就用id区分而已
      • Jafir:希望多来一些例子和应用场景
        Tamic:@Jafir 官方文档有加载已安装app列表的列子
      • Jafir:哈哈 又学到一个新知识。之前从未用过呢?请问这样的应用场景多么?可不可以多举一些例子,就像你上面举得 同时操作电话本,一个改动了数据,另一个也需要根据数据的更改而被通知进行数据更新
        Tamic:@Jafir 多 只要你想玩
      • Tamic:@崔小波 AsyncTaskLoder自带轮询机制,并开放了doInBackground让子类必须实现,如果是通过网络访问的加载数据源,是没必要做到实时 如果你非要实时那么和推送没什么区别,单纯的AsyncTaskLoder是做不到实时更新。如果你要做到实时,需要自我实现监听,不断心跳去请求服务器,这样不仅耗电而且耗流量,一般我们实时大多用在数据库和内容提供者的操作上,比如两个App同时可以对电话本操作,一个app写入一个联系人,这时内容提供者会告诉cusurLeader进行有数据更新了,当然另一个app也就实时加载联系人了
      • 8314e3a0c30e:没想到LoaderManager还这么强大,看完感觉 handler + thread确实很low!
        Tamic:@赵丰年 是的 最新的源码内部很多地方用Loader
      • cuixbo:我们该如何去理解 AsyncTaskLoader 会自动刷新数据变化?最主要是加载数据,使用loader我们无须关注数据何时改变了?结合访问网络数据该如何理解?

      本文标题:Android 深入理解Loader机制 让APP轻装上阵

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