Android异步处理之AsyncTaskLoader简单使用

2023-12-07 13:49

本文主要是介绍Android异步处理之AsyncTaskLoader简单使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介

不管是在Android应用开发还是Android平台开发中,异步处理通常是最基本的coding要求。如果你还在主线程中写一些数据库,网络请求,读写本地文件等操作的话那说明你还不是一个合格的Android程序员。

通常情况下我们使用的最多的Android异步处理方法是AsyncTaskHandler,但今天要给大家带来的是大家不常使用的AsyncTaskLoader的使用方法。

AsyncTaskLoader从名字看出来它似乎和AsyncTask有关系,看一下Google官方是怎么给AsyncTaskLoader下定义的:

Abstract Loader that provides an AsyncTask to do the work

果然是样,AsyncTaskLoader是使用一个AsyncTask来进行异步处理的。那么问题来了,既然都有了AsyncTask了为什么还要搞出来一个AsyncTaskLoader呢?

其实AsyncTaskLoader远没有大家想的那么简单。说的通俗一点,如果把AsyncTask比作一台烤面包机的话,那么AsyncTaskLoader就是操作烤面包机的面包师。AsyncTask如同烤面包机接受命令完成面包的烤制任务,一旦任务完成它就停止了工作。然而AsyncTaskLoader如同面包师一样要根据顾客的需求来使用烤面包机。顾客会不停的光顾,那么面包师就会不停的使用烤面包机烤面包。

具体事例

下面我们就通过烤面包机和面包师的例子来演示一下AsyncTaskLoader的使用方法。

首先肯定少不了面包师(Baker):

package com.example.asyncloaderdemo;import java.util.ArrayList;
import java.util.List;import android.content.AsyncTaskLoader;
import android.content.Context;public class Baker extends AsyncTaskLoader<List<Bread>> {// 用于查询当前需要多少个面包BakeryCallback mCallback;//面包房回调,用于获得当面面包需求量interface BakeryCallback {int getNeededBreads();}public Baker(Context context, BakeryCallback callback) {super(context);mCallback = callback;}@Overridepublic List<Bread> loadInBackground() {List<Bread> breads = new ArrayList<Bread>();//获得当前需要做的面包int needs = mCallback.getNeededBreads();for (int i = 0; i < needs; i++) {//制作面包,耗时操作breads.add(new Bread());}//面包制作完成return breads;}@Overridepublic void deliverResult(List<Bread> data) {super.deliverResult(data);}@Overrideprotected void onStartLoading() {forceLoad();}@Overrideprotected void onStopLoading() {cancelLoad();}@Overrideprotected void onReset() {super.onReset();}
}
  • 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
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 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
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57

面包师有了,面包房(Bakery)也不能少

package com.example.asyncloaderdemo;import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;public class Bakery extends BroadcastReceiver {final Baker mBaker;public static String CUSTOMER_ACTION = "com.example.asyncloaderdemo.new_customer" ;public Bakery(Baker baker) {mBaker = baker;IntentFilter filter = new IntentFilter(CUSTOMER_ACTION);baker.getContext().registerReceiver(this, filter);}@Override public void onReceive(Context context, Intent intent) {//通知面包师来客人了,要做面包了。mBaker.onContentChanged();}}
  • 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
  • 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

面包房和面包师都有了,还缺一个场景(MainActivity)

package com.example.asyncloaderdemo;import java.util.ArrayList;
import java.util.List;
import java.util.Random;import com.example.asyncloaderdemo.Baker.BakeryCallback;import android.support.v7.app.ActionBarActivity;
import android.app.LoaderManager.LoaderCallbacks;
import android.content.Intent;
import android.content.Loader;
import android.os.Bundle;
import android.util.Log;public class MainActivity extends ActionBarActivity {private LoaderCallbacks<List<Bread>> mCallbacks;//面包房private Bakery mBakery;//面包师private Baker mBaker;//面包需求量private int mNeededBreads;//唯一标识private final int mLoaderId = 42;private BakeryCallback mBreadCallback = new BakeryCallback() {@Overridepublic int getNeededBreads() {return mNeededBreads;}};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mNeededBreads = 0;mBaker = new Baker(this, mBreadCallback);mBakery = new Bakery(mBaker);mCallbacks = new LoaderCallbacks<List<Bread>>() {@Overridepublic Loader<List<Bread>> onCreateLoader(int id, Bundle args) {if (mBaker == null) {mBaker = new Baker(MainActivity.this, mBreadCallback);}return mBaker;}@Overridepublic void onLoadFinished(Loader<List<Bread>> loader, List<Bread> data) {mNeededBreads = 0 ;//面包师完成面包烤制Log.d("scott", "sell " + data.size() + " breads") ;}@Overridepublic void onLoaderReset(Loader<List<Bread>> loader) {}};//面包师开始工作getLoaderManager().restartLoader(mLoaderId, null, mCallbacks);//顾客开始上门mockCustomer();}@Overrideprotected void onDestroy() {super.onDestroy();unregisterReceiver(mBakery);}//模拟源源不断的顾客需求private void mockCustomer(){new Thread(new Runnable() {@Overridepublic void run() {while(true){try {Thread.sleep(3000);Random random = new Random();mNeededBreads =random.nextInt(10);Intent intent = new Intent(Bakery.CUSTOMER_ACTION);sendBroadcast(intent);} catch (InterruptedException e) {e.printStackTrace();}}}}).start();}
}
  • 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
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 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
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95

接下来我们来看一下程序运行结果: 
面包房运行结果 
通过上图并结合代码可以看出来每隔三秒就会有新的顾客上门,顾客上门后面包房通知面包师需要做面吧哦了,接着面包师就会在后台不停的开始使用面包机(AsyncTask)做面包。

总结

说到这里可能有些同学有疑问了,我怎么从头到尾都没有开到AsyncTask的影子呢?你当然看不到,这就是AsyncTaskLoader设计精妙之处,它做到了让你唯一需要考虑的就是烤面包(异步处理)这个事物逻辑,而不需要考虑异步处理本身的实现上。同时这也充分体现了设计模式中的单一职责最少知道原则。

使用场景

AsyncTaskLoader一般使用在数据源处于不断更新并且请求刷新数据源是个耗时操作的情况下还需要UI去同步更新相关数据的场景(这句话怎么这么拗口)。

这篇关于Android异步处理之AsyncTaskLoader简单使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中的ConcurrentBitSet使用小结

《Java中的ConcurrentBitSet使用小结》本文主要介绍了Java中的ConcurrentBitSet使用小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、核心澄清:Java标准库无内置ConcurrentBitSet二、推荐方案:Eclipse

Go语言结构体标签(Tag)的使用小结

《Go语言结构体标签(Tag)的使用小结》结构体标签Tag是Go语言中附加在结构体字段后的元数据字符串,用于提供额外的属性信息,这些信息可以通过反射在运行时读取和解析,下面就来详细的介绍一下Tag的使... 目录什么是结构体标签?基本语法常见的标签用途1.jsON 序列化/反序列化(最常用)2.数据库操作(

Java中ScopeValue的使用小结

《Java中ScopeValue的使用小结》Java21引入的ScopedValue是一种作用域内共享不可变数据的预览API,本文就来详细介绍一下Java中ScopeValue的使用小结,感兴趣的可以... 目录一、Java ScopedValue(作用域值)详解1. 定义与背景2. 核心特性3. 使用方法

spring中Interceptor的使用小结

《spring中Interceptor的使用小结》SpringInterceptor是SpringMVC提供的一种机制,用于在请求处理的不同阶段插入自定义逻辑,通过实现HandlerIntercept... 目录一、Interceptor 的核心概念二、Interceptor 的创建与配置三、拦截器的执行顺

Python字符串处理方法超全攻略

《Python字符串处理方法超全攻略》字符串可以看作多个字符的按照先后顺序组合,相当于就是序列结构,意味着可以对它进行遍历、切片,:本文主要介绍Python字符串处理方法的相关资料,文中通过代码介... 目录一、基础知识:字符串的“不可变”特性与创建方式二、常用操作:80%场景的“万能工具箱”三、格式化方法

Spring Boot 处理带文件表单的方式汇总

《SpringBoot处理带文件表单的方式汇总》本文详细介绍了六种处理文件上传的方式,包括@RequestParam、@RequestPart、@ModelAttribute、@ModelAttr... 目录方式 1:@RequestParam接收文件后端代码前端代码特点方式 2:@RequestPart接

C#中checked关键字的使用小结

《C#中checked关键字的使用小结》本文主要介绍了C#中checked关键字的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录✅ 为什么需要checked? 问题:整数溢出是“静默China编程”的(默认)checked的三种用

C#中预处理器指令的使用小结

《C#中预处理器指令的使用小结》本文主要介绍了C#中预处理器指令的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 第 1 名:#if/#else/#elif/#endif✅用途:条件编译(绝对最常用!) 典型场景: 示例

Mysql中RelayLog中继日志的使用

《Mysql中RelayLog中继日志的使用》MySQLRelayLog中继日志是主从复制架构中的核心组件,负责将从主库获取的Binlog事件暂存并应用到从库,本文就来详细的介绍一下RelayLog中... 目录一、什么是 Relay Log(中继日志)二、Relay Log 的工作流程三、Relay Lo

使用Redis实现会话管理的示例代码

《使用Redis实现会话管理的示例代码》文章介绍了如何使用Redis实现会话管理,包括会话的创建、读取、更新和删除操作,通过设置会话超时时间并重置,可以确保会话在用户持续活动期间不会过期,此外,展示了... 目录1. 会话管理的基本概念2. 使用Redis实现会话管理2.1 引入依赖2.2 会话管理基本操作