使用Loader加载异步数据

2024-04-28 11:08

本文主要是介绍使用Loader加载异步数据,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Loader与LoaderManager

Loader设计用于从数据源加载某类数据(如对象)。数据源可以是磁盘、数据库、ContentProvider、网络或者另一进程。Loader可在不阻塞主线程的情况下获取并发送结果数据给接收者。

loader有三种内置类型:Loader、AsyncTaskLoader和CursorLoader。作为基类,Loader本身没有多大用处。它定义了供LoaderManager与其它loader通讯时使用的API。

AsyncTaskLoader是一个抽象Loader。它使用AsyncTask将数据加载任务转移到其他线程上处理。几乎所有我们创建的有用loader类都是AsyncTaskLoader的一个子类。

CursorLoader通过继承AsyncTaskLoader类,它借助于ContentResolver从ContentProvider加载Cursor。

LoaderManager管理着与loader间的所有通讯,并负责启动、停止和管理与组件关联的Loader生命周期方法。在Activity或Fragment中,我们可以使用getLoaderManager()方法返回一个实例并与之交互。

要初始化Loader,可使用initLoader(int, Bundle, LoaderCallbacks)方法。该方法中的三个参数,第一个是整数类型的loader标识符,第二个是Bundle参数(值可为空),最后一个是LoaderCallbacks接口的实现。

要强制重启现有loader,可使用restartLoader(int, Bundle, LoaderCallbacks)方法。在明确知道或怀疑数据比较陈旧时,我们通常会使用该方法重新加载最新数据。

为什么要实用loader而不直接使用AsyncTask呢?一个最有力说服力的理由是,因设备旋转等原因发生配置更改时,LoaderManager可保证组件的loader及其数据不会丢失。

如果使用AsyncTask加载数据,配置发生改变时,我们必须亲自管理其生命周期并保存它获取的数据。虽然我们经常使用保留的Fragment(使用setRetainInstance(true)方法)解决了这些麻烦问题,但某些场景下,我们还是必须亲自介入并编写对应代码,才能保证一切尽在掌握。

loader的设计目的就是要解决部分(不是全部)这样的恼人问题。配置发送改变后,如果我们初始化一个已完成数据加载的loader,它会立即发送取得的数据,而不是尝试再次获取数据。而且,无论fragment是否保留,它都是这样工作的。这样一来,无需再考虑保留fragment带来的生命周期难题。我们的开发工作也会因此轻松很多。

使用Loader

新建SQLiteCursorLoader 类
package com.huangfei.runtracker;import android.content.Context;
import android.database.Cursor;
import android.support.v4.content.AsyncTaskLoader;
/***SQLiteCursorLoader是CursorLoader框架类的简化版本,可与来自任何数据源的Cursor协同工作.*该类复制了CursorLoader类的大部分代码,但并不需要使用ContentProvider类。*/
public abstract class SQLiteCursorLoader extends AsyncTaskLoader<Cursor> {private Cursor mCursor;//SQLiteCursorLoader可高效加载Cursor,并将其保存在mCursor实例变量中public SQLiteCursorLoader(Context context) {super(context);}protected abstract Cursor loadCursor();@Overridepublic Cursor loadInBackground() {Cursor cursor = loadCursor();if(cursor != null){//保证数据在发送给主线程之前已加载到内存中cursor.getCount();}return cursor;}/*** deliverResult(Cursor)方法负责两件事。* 如果load启动了(这表示数据可以发送了),超类版本的deliverResult(Cursor)方法会被调用。* 如果旧的cursor不再需要,它会被关闭以及释放资源。现有的cursor可能会被缓存或重新发送,因此在* 关闭旧的cursor前,必须确认新旧cursor并不相同。*/@Overridepublic void deliverResult(Cursor data) {Cursor oldCursor = mCursor;mCursor = data;if(isStarted()){super.deliverResult(data);}if(oldCursor != null && oldCursor != data && !oldCursor.isClosed()){oldCursor.close();}}@Overrideprotected void onStartLoading() {if(mCursor != null){deliverResult(mCursor);}if(takeContentChanged() || mCursor == null){forceLoad();}}@Overrideprotected void onStopLoading() {cancelLoad();}@Overridepublic void onCanceled(Cursor data) {if(data != null && !data.isClosed()){data.close();}}@Overrideprotected void onReset() {super.onReset();onStopLoading();if(mCursor != null && !mCursor.isClosed()){mCursor.close();}mCursor = null;}
}
新建DateLoader 类
package com.huangfei.runtracker;import android.content.Context;
import android.support.v4.content.AsyncTaskLoader;/*** DateLoader可加载任意类型数据,并能帮助子类简化使用AsyncTaskLoader。* DateLoader类可处理一些任何AsyncTaskLoader子类都能处理的简单任务,而仅把loadInBackground()方法的实现任务留给了自己的子类。* DateLoader使用D泛型类存储加载的数据实例。*/public abstract class DateLoader<D> extends AsyncTaskLoader<D> {private D mDate;public DateLoader(Context context) {super(context);}/*** deliverResult(D)方法先将新数据对象存储起来,然后如果loader以启动,则调用超类的版本的方法将数据发送出去。*/@Overridepublic void deliverResult(D data) {mDate = data;if (isStarted()) {super.deliverResult(data);}}/*** 检查数据是否已加载,如已加载则立即发送;否则,就调用超类的forceLoad()方法去获取数据。*/@Overrideprotected void onStartLoading() {if (mDate != null) {deliverResult(mDate);} else {forceLoad();}}}
在Activity或Fragment中调用AsyncTaskLoader
getLoaderManager().initLoader(0, null, this);...private static class RunListCursorLoader extends SQLiteCursorLoader{public RunListCursorLoader(Context context) {super(context);}@Overrideprotected Cursor loadCursor() {return RunManager.get(getContext()).queryRuns();}}/*** 需要创建loader时,LoaderManager会调用onCreateLoader(int, Bundle)方法。如果有多个相同类型的loader,* 可使用id参数区分它们。而Bundle会保存传入的任何类型参数*/@Overridepublic Loader<Cursor> onCreateLoader(int id, Bundle args) {return new RunListCursorLoader(getActivity());}/*** 数据在后台一完成加载,onLoadFinished(Loader<Cursor>, Cursor)方法就会在主线程上被调用*/@Overridepublic void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {RunCursorAdapter adapter = new RunCursorAdapter(getActivity(), (RunCursor)cursor);setListAdapter(adapter);}/*** 在没有可加载数时,onLoaderReset(Loader<Cursor>)方法会被调用,安全起见,我们设置列表adapter值为空,以停止使用cursor。*/@Overridepublic void onLoaderReset(Loader<Cursor> loader) {setListAdapter(null);}

代码地址

这篇关于使用Loader加载异步数据的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/943103

相关文章

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

MyBatis-Plus通用中等、大量数据分批查询和处理方法

《MyBatis-Plus通用中等、大量数据分批查询和处理方法》文章介绍MyBatis-Plus分页查询处理,通过函数式接口与Lambda表达式实现通用逻辑,方法抽象但功能强大,建议扩展分批处理及流式... 目录函数式接口获取分页数据接口数据处理接口通用逻辑工具类使用方法简单查询自定义查询方法总结函数式接口

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

使用Python删除Excel中的行列和单元格示例详解

《使用Python删除Excel中的行列和单元格示例详解》在处理Excel数据时,删除不需要的行、列或单元格是一项常见且必要的操作,本文将使用Python脚本实现对Excel表格的高效自动化处理,感兴... 目录开发环境准备使用 python 删除 Excphpel 表格中的行删除特定行删除空白行删除含指定

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

prometheus如何使用pushgateway监控网路丢包

《prometheus如何使用pushgateway监控网路丢包》:本文主要介绍prometheus如何使用pushgateway监控网路丢包问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录监控网路丢包脚本数据图表总结监控网路丢包脚本[root@gtcq-gt-monitor-prome

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数