AsyncQueryHandler简介:

异步的查询操作帮助类,可以处理增删改(ContentProvider提供的数据)



使用场景:

在一般的应用中可以使用ContentProvider去操作数据库

这在数据量很小的时候是没有问题的,但是如果数据量大了,可能导致UI线程发生ANR异常(超过5秒)。

当然你也可以写个Handler去做这些操作,只是你每次使用ContentProvider时都要再写个Handler,必然降低了效率。

因此当数据量较大时,最好还是使用Android已经封装好的异步查询框架AsyncQueryHandler,优化我们的代码.

要注意的是,一般在查询本地的应用的数据的时候要去采用CursorAdapter

AsyncQueryHandler内部实现

AsyncQueryHandler类封装了调用者线程与工作线程的交互过程。交互的主体是两个Handler,一个运行在调用者线程中,一个运行在工作者线程中。通过提供onXXXComplete的回调接口,实现事件的完成处理。



API中提供

startInsert,

startDelete,

startUpdate,

startQuery四种方法,并有响应的onXXXComplete()方法.于对应的4个onXXXComplete()方法都是空实现,因此我们完成相应调用后进行后续其他的操作.

使用方法

继承AsyncQueryHandler类,并提供onXXXComplete方法的实现(可以实现任何一个或多个,当然你也可以一个也不实现,如果你不关注操作数据库的結果),在你的实现中做一些对数据库操作完成的处理。

使用时直接调用startXXX方法即可。传入的通用参数如下:

token,一个令牌,主要用来标识查询,保证唯一即可.需要跟onXXXComplete方法传入的一致。(当然你也可以不一致,同样在数据库的操作结束后会调用对应的onXXXComplete方法 )。当你使用很多次查询的时候,这个值就相当于绑定了你的查询方法。

cookie,你想传给onXXXComplete方法使用的一个对象。(没有的话传递null即可)

Uri uri(进行查询的通用资源标志符):

projection 查询的列

selection  限制条件

selectionArgs 查询参数

orderBy 排序条件

下面简单介绍一下查询操作,如果查询操作熟悉了,其他的也就迎刃而解了。

例如,我要查询一下我手机里面有多少视频文件,以及对应的视频文件的名称、视频占用内存、视频时长,以及路径。按照以往的方式,我们会这么写:

 List<VideoInfo> videoDatas = new ArrayList<VideoInfo>();

		/*//参数:url、查询哪些列、选择条件占位符、选择条件占位符?用什么替换、查询后排序方式
Cursor cursor = context.getContentResolver().query(Media.EXTERNAL_CONTENT_URI,
new String[]{Media.TITLE,//视频名称
Media.SIZE,Media.DURATION,Media.DATA},//大小、时长、视频路径
null, null, null);
while(cursor.moveToNext()){
VideoInfo videoInfo = new VideoInfo();
String title = cursor.getString(cursor.getColumnIndex(Media.TITLE));
int size = cursor.getInt(cursor.getColumnIndex(Media.SIZE));
int duration = cursor.getInt(cursor.getColumnIndex(Media.DURATION));
String path = cursor.getString(cursor.getColumnIndex(Media.DATA));; videoInfo.setDuration(duration);
videoInfo.setPath(path);
videoInfo.setSize(size);
videoInfo.setTitle(title);
videoDatas.add(videoInfo);
}

很显然,以前的方式就是内容提供者查询数据库操作。外界如果想要获取这些信息,是需要开启Handler---Thread的,如下:

new Thread(new Runnable() {

			@Override
public void run() {
// 获取数据
videDatas = MediaInfoEngine.getVideoInfo(UIUtils.getContext());
UIUtils.runOnUiThread(new Runnable() { @Override
public void run() {
adapter.notifyDataSetChanged();
}
});
}
}).start();

这样就获取到了手机所有视频文件信息了。但是,如果你手机里有1万条视频文件,这个时候,即使使用这种方式,估计上边业报ANR了。这个时候,AsyncQueryHandler帮我们解决很多问题,因为它已经自动帮我们封装了子线程。提供子线程DB操作,然后回调到主线程UI操作。写法如下:

AsyncQueryHandler queryHandler = new AsyncQueryHandler(context.getContentResolver()) {
/**查询的响应回调*/
@Override
protected void onQueryComplete(int token, Object cookie,
Cursor cursor) {
if(token == 1){
while(cursor.moveToNext()){
VideoInfo videoInfo = new VideoInfo();
String title = cursor.getString(cursor.getColumnIndex(Media.TITLE));
int size = cursor.getInt(cursor.getColumnIndex(Media.SIZE));
int duration = cursor.getInt(cursor.getColumnIndex(Media.DURATION));
String path = cursor.getString(cursor.getColumnIndex(Media.DATA));; videoInfo.setDuration(duration);
videoInfo.setPath(path);
videoInfo.setSize(size);
videoInfo.setTitle(title);
videoDatas.add(videoInfo);
}
}
}
}; /**启动异步查询*/
//参数一:请求码
queryHandler.startQuery(1, null, Media.EXTERNAL_CONTENT_URI,
new String[]{Media.TITLE,//视频名称
Media.SIZE,Media.DURATION,Media.DATA},//大小、时长、视频路径
null, null, null);

具体的参数,在开篇就介绍的很详细了。简单说一下如何操作的:首先执行开始查询操作startQuary(),这个方法给我们封装好了,是在子线程执行的,在它内部自定查询完毕后会生成一个curssor对象,回调内部类里面的onQueryComplete方法,这个方法的三个参数分别是:1:请求码(可以想一下intent隐式启动的那个请求码的意思)2:执行startQuary()时候,第二参数。一般传入null,但是有时候会是某个对象(场景:AsyncQueryHandler与CursorAdapter结合使用的时候,这里就可以传入一个curssorAdapter对象,在回到方法里面就可以接收这个对象。回想message的obj字段)3:startQuary()方法查询数据库后所有返回的结果游标。接下来就很简单了,遍历cursor保存数据即可了。

这个时候,我们获取数据的时候,也就不用自己开启线程了。直接获取即可,如下:

 // 获取数据,不用再开启你子线程了
videDatas = MediaInfoEngine.getVideoInfo(UIUtils.getContext());
adapter.notifyDataSetChanged();

这里我没有用到CursorAdapter,因为实际开发中,也没怎么用到这个适配器。后续博客会再对这个适配器做详细的介绍。

Android 异步查询框架AsyncQueryHandler的使用的更多相关文章

  1. Android 异步开发之 AsyncQueryHandler 批量添加联系人

    AsyncQueryHandler: 官方解释是一个异步帮助类(A helper class to help make handling asynchronous ContentResolver qu ...

  2. Android异步任务处理框架AsyncTask源代码分析

    [转载请注明出处:http://blog.csdn.net/feiduclear_up CSDN 废墟的树] 引言 在平时项目开发中难免会遇到异步耗时的任务(比方最常见的网络请求).遇到这样的问题.我 ...

  3. android异步Http框架

    首先在GitHub上下载异步Http框架代码以及相关文档: 将jar包放入lib包中即可: 接下来分别实现get.post.文件上传功能实现: 代码实现如下: AsyncHttpClient clie ...

  4. [android] 异步http框架与实现原理

    介绍github上的异步http框架android-async-http loopj开发 获取AsyncHttpClient对象,通过new 调用AsyncHttpClient对象的get(url,r ...

  5. Android 异步Http框架简介和实现原理

    在前几篇文章中<Android 采用get方式提交数据到服务器><Android 采用post方式提交数据到服务器><Android 采用HttpClient提交数据到服 ...

  6. Android 从零开始打造异步处理框架

    转载请标明出处:http://www.cnblogs.com/zhaoyanjun/p/5995752.html 本文出自[赵彦军的博客] 概述 在Android中会使用异步任务来处理耗时操作,避免出 ...

  7. android 学习随笔十二(网络:使用异步HttpClient框架)

    使用异步HttpClient框架发送get.post请求 在https://github.com/ 搜索 asyn-http https://github.com/search?utf8=✓& ...

  8. Android Native层异步消息处理框架

     *本文系作者工作学习总结,尚有不完善及理解不恰当之处,欢迎批评指正* 一.前言 在NuPlayer中,可以发现许多类似于下面的代码: //============================== ...

  9. 各种Android UI开源框架 开源库

    各种Android UI开源框架 开源库 转 https://blog.csdn.net/zhangdi_gdk2016/article/details/84643668 自己总结的Android开源 ...

随机推荐

  1. [LeetCode] Convert BST to Greater Tree 将二叉搜索树BST转为较大树

    Given a Binary Search Tree (BST), convert it to a Greater Tree such that every key of the original B ...

  2. NC二次开发常用的表

    常用的表: 收费清单:pr_cr_receivables 会计月份: bd_accperiodmonth 20180416

  3. .NET CORE 2.0之 依赖注入在类中获取IHostingEnvironment,HttpContext

    在.NET CORE 中,依赖注入非常常见, 在原先的 HttpContext中常用的server.Mappath已经么有了如下: HttpContext.Current.Server.MapPath ...

  4. 计蒜客NOIP2017提高组模拟赛(五)day2-蚂蚁搬家

    传送门 这题可以用线段树来维护 #include<cstdio> #include<cstdlib> #include<algorithm> #include< ...

  5. bzoj3825 NOI2017 游戏

    题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行nn 场游戏,每场游戏使用一张地 ...

  6. hdu 5583 Kingdom of Black and White

    Kingdom of Black and White Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Ja ...

  7. VS2012中C++,#include无法打开自己所写的头文件(.h)

    最近刚开始学cocos2d-x,创建项目之后,自己按照<cocos2d-x 3.x 游戏开发>的教程写代码 先写了一个头文件  MyHelloWorldScene.h 然后在  AppDe ...

  8. js生成四位随机数的简便方法

    do out = Math.floor(Math.random()*10000); while( out < 1000 ) alert( out );

  9. Mybatis自动生成实体类和实体映射工具

    Mybatis Mysql生成实体类 用到的Lib包: mybatis-generator-core-1.3.2.jarmysql-connector-java-5.1.30.jar 1. 创建一个文 ...

  10. 去除html标记和替换script标记

    1: /// <summary> 2: /// 去除HTML标记 3: /// </summary> 4: /// <param name="NoHTML&qu ...