从最简单的图片加载,教你Android实现异步!

简介: 异步,在安卓开发中简直是再熟悉不过了。说到异步,脑海中立马浮现的就是多线程开发,Thread、Handler啥的一一涌上心头…我们知道在Android开发中不能在非UI线程中更新UI,但是,有的时候我们需要在代码中执行一些诸如访问网络、查询数据库等耗时操作,为了不阻塞UI线程,我们时常会开启一个新的线程(工作线程)来执行这些耗时操作,然后我们可能需要将查询到的数据渲染到UI组件上,那么这个时候我们就需要考虑异步更新UI的问题了。

异步,在安卓开发中简直是再熟悉不过了

说到异步,脑海中立马浮现的就是多线程开发,Thread、Handler啥的一一涌上心头…

我们知道在Android开发中不能在非UI线程中更新UI,但是,有的时候我们需要在代码中执行一些诸如访问网络、查询数据库等耗时操作,为了不阻塞UI线程,我们时常会开启一个新的线程(工作线程)来执行这些耗时操作,然后我们可能需要将查询到的数据渲染到UI组件上,那么这个时候我们就需要考虑异步更新UI的问题了。

今天我们从一个简单的业务需求,给大家介绍几种实现异步的方式,最后两个简直爽到不行。

业务是这样的:需要根据文件地址,加载本地图片,最后在ImageView上显示。当然了,从文件中加载图片,是一个耗时操作,必须在子线程中执行,ImageView显示图片呢,又属于UI操作,需要回到主线程。接下来列举几种实现方式:

Thread+Handler

使用Thread+Handler是最传统的实现异步方式了,看下代码:

 new Thread(new Runnable() {
 @Override
 public void run() {
 Bitmap bitmap = getBitmapFromFile(PATH);
 handler.post(new Runnable() {
 @Override
 public void run() {
 imageView.setImageBitmap(bitmap);
 }
 });
 }
 }).start();

如果熟悉Lambda表达式的话,也可以这样写:

 new Thread(() -> {
 Bitmap bitmap = getBitmapFromFile(PATH);
 handler.post(() -> imageView.setImageBitmap(bitmap));
 }).start();

这样看来代码干净了许多。

除了实现Runnable,还可以继承Thread,实现run方法来做到开启子线程。但由于Java的单继承多实现,所以还是使用实现Runnable方式更实用一些。handler的post方法可以将消息发送回主线程,以实现线程间切换。

这种方式在需要的地方new一个对象,使得代码繁乱,不易管理,对系统资源也不便管理。

AsyncTask

AsyncTask提供了方便的接口实现工作线程和主线程的通信。先贴代码:

 class BitmapAsyncTask extends AsyncTask<String, Integer, Bitmap> {
 @Override
 protected void onProgressUpdate(Integer... values) {
 super.onProgressUpdate(values);
 }
 @Override
 protected Bitmap doInBackground(String... strings) {
 // 在doInBackground方法中执行耗时操作
 Bitmap bitmap = getBitmapFromFile(strings[0]);
 return bitmap;
 }
 @Override
 protected void onPostExecute(Bitmap bitmap) {
 super.onPostExecute(bitmap);
 // 在onPostExecute方法中进行ui操作
 imageView.setImageBitmap(bitmap);
 }
 }
new BitmapAsyncTask().execute(PATH);

AsyncTask就是一个封装过的后台任务类,顾名思义就是异步任务。AsyncTask定义了三种泛型类型 Params,Progress和Result。

doInBackground(Params…) 后台执行,比较耗时的操作都可以放在这里。注意这里不能直接操作UI。此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用publicProgress(Progress…)来更新任务的进度。

onPostExecute(Result) 相当于Handler 处理UI的方式,在这里面可以使用在doInBackground 得到的结果处理操作UI。 此方法在主线程执行,任务执行的结果作为此方法的参数返回。

这种方式使用了线程池+Handler实现,较好得管理分配资源,还可以拿到进度回调,有较高的拓展性。但需要创建新类,代码也会随之增加,对于简单的异步操作,这种方式有些繁琐。

RxJava

主要还是用到了RxJava的Scheduler(调度器)来实现线程切换,看下代码:

 Observable observable = Observable.create(new Observable.OnSubscribe<Bitmap>() {
 @Override
 public void call(Subscriber<? super Bitmap> subscriber) {
 Bitmap bitmap = getBitmapFromFile(PATH);
 subscriber.onNext(bitmap);
 }
 });
 observable.subscribeOn(Schedulers.io()) // 指定 subscribe() 发生在 IO 线程
 .observeOn(AndroidSchedulers.mainThread()) // 指定 Subscriber 的回调发生在主线程
 .subscribe(new Subscriber<Bitmap>() {
 @Override
 public void onCompleted() {
 }
 @Override
 public void onError(Throwable e) {
 }
 @Override
 public void onNext(Bitmap bitmap) {
 imageView.setImageBitmap(bitmap);
 }
 });

使用Observable.create创建Observable,在call方法中进行耗时操作,执行完成后发送消息,在观察者中的onNext中处理。

使用subscribeOn和observeOn进行线程切换。

使用RxJava的好处是很轻松得实现线程切换,还可以指定线程,有异常捕获机制。但对于不熟悉RxJava的朋友来说会有些…

Kotlin协程

最后要安利一个非常酷炫的方式,那就是Kotlin协程。

越来越多的公司和项目开始使用Kotlin编码,毕竟Kotlin得到了谷歌爸爸的支持,而且Kotlin的优秀语言特性,使得它受到开发者的广泛欢迎。

今天介绍Kotlin的一个概念,叫做协程。协程是由程序直接实现的,是一种轻量级线程,kotlin也为此提供了标准库和额外的实验库。标准库为kotlin.coroutines.experimental(写作时使用kotlin-1.20版本),可见仍然还是一个实验性功能。

看下代码

先定义一个后台CoroutineContext,协程上下文,很容易理解,就是执行环境。

val Background = newFixedThreadPoolContext(2, "bg")
mWriteJob = launch(Background) {
 var bitmap = getBitmapFromFile(PATH);
 launch(UI){
 imageView.setImageBitmap(bitmap)
 }
 }

最后会返回一个Job对象,可以调用方法将其任务停止:

 if (mWriteJob != null && mWriteJob!!.isActive) {
 mWriteJob!!.cancel()
 }

不由得想感叹一下,使用协程做轻量的异步操作,简直爽到不行。

但毕竟协程可能还是了解不多,不免会有一些坑的出现,但多去了解和使用,想必也是很酷的。

小结

从个人感觉来说,我比较推荐使用RxJava和协程来实现,处理周密的话,轻松避免资源浪费和内存泄漏。

Android中的异步操作,实现方式有好多种,各有利弊,就需要我们针对具体业务需求来选择合适的方式,使得功能完成的前提下,优化性能,优化代码。

给看到最后的朋友们发一波福利;

现在加Android开发群;701740775,可免费领取一份最新Android高级架构技术体系大纲和视频资料,以及五年积累整理的所有面试资源笔记。加群请备注csdn领取高级大纲。

相关文章
|
7月前
|
Java 数据库 Android开发
Android异步之旅:探索AsyncTask
Android异步之旅:探索AsyncTask
62 0
|
3月前
|
存储 缓存 编解码
Android经典面试题之图片Bitmap怎么做优化
本文介绍了图片相关的内存优化方法,包括分辨率适配、图片压缩与缓存。文中详细讲解了如何根据不同分辨率放置图片资源,避免图片拉伸变形;并通过示例代码展示了使用`BitmapFactory.Options`进行图片压缩的具体步骤。此外,还介绍了Glide等第三方库如何利用LRU算法实现高效图片缓存。
72 20
Android经典面试题之图片Bitmap怎么做优化
|
7月前
|
Android开发
Android通过手势(多点)缩放和拖拽图片
Android通过手势(多点)缩放和拖拽图片
57 4
|
7月前
|
Java Android开发
android 下载图片的问题
android 下载图片的问题
50 3
|
4月前
|
数据处理 开发工具 数据安全/隐私保护
Android平台RTMP推送|轻量级RTSP服务|GB28181接入之文字、png图片水印的精进之路
本文探讨了Android平台上推流模块中添加文字与PNG水印的技术演进。自2015年起,为了满足应急指挥及安防领域的需求,逐步发展出三代水印技术:第一代为静态文字与图像水印;第二代实现了动态更新水印内容的能力,例如实时位置与时间信息;至第三代,则优化了数据传输效率,直接使用Bitmap对象传递水印数据至JNI层,减少了内存拷贝次数。这些迭代不仅提升了用户体验和技术效率,也体现了开发者追求极致与不断创新的精神。
|
4月前
|
自然语言处理 定位技术 API
Android经典实战之如何获取图片的经纬度以及如何根据经纬度获取对应的地点名称
本文介绍如何在Android中从图片提取地理位置信息并转换为地址。首先利用`ExifInterface`获取图片内的经纬度,然后通过`Geocoder`将经纬度转为地址。注意操作需在子线程进行且考虑多语言支持。
269 4
|
4月前
|
XML 前端开发 Android开发
Android经典实战之Kotlin中实现圆角图片和圆形图片
本文介绍两种实现圆角图像视图的方法。第一种是通过自定义Kotlin `AppCompatImageView`,重写`onDraw`方法使用`Canvas`和`Path`进行圆角剪裁。第二种利用Android Material库中的`ShapeableImageView`,简单配置即可实现圆角效果。两种方法均易于实现且提供动态调整圆角半径的功能。
92 0
|
6月前
|
存储 Java 数据库连接
Android Java开发异步
【6月更文挑战第15天】
|
6月前
|
JSON Java API
【Android】使用 Retrofit2 发送异步网络请求的简单案例
**摘要:** Retrofit是Android和Java的HTTP客户端库,简化了RESTful API交互。它通过Java接口定义HTTP请求,并提供注解管理参数、HTTP方法等。要使用Retrofit,首先在AndroidManifest.xml中添加`INTERNET`权限,然后在`build.gradle`中引入Retrofit和Gson依赖。创建服务器响应数据类和描述接口的接口,如`Result`和`Api`。通过Retrofit.Builder配置基础URL并构建实例,之后调用接口方法创建Call对象并发送异步请求。
234 1
|
6月前
|
Android开发 Kotlin
Android面试题 之 Kotlin DataBinding 图片加载和绑定RecyclerView
本文介绍了如何在Android中使用DataBinding和BindingAdapter。示例展示了如何创建`MyBindingAdapter`,包含一个`setImage`方法来设置ImageView的图片。布局文件使用`&lt;data&gt;`标签定义变量,并通过`app:image`调用BindingAdapter。在Activity中设置变量值传递给Adapter处理。此外,还展示了如何在RecyclerView的Adapter中使用DataBinding,如`MyAdapter`,在子布局`item.xml`中绑定User对象到视图。关注公众号AntDream阅读更多内容。
105 1