AsyncTask 使用及封装实践(1)

private class MyDownloadTask extends AsyncTask<Void, FileInfo, FileInfo>{

}

那Void, FileInfo, FileInfo,这几个参数的类型在哪里体现出来呢?

请看下面注释

private class MyDownloadTask extends AsyncTask<Void, FileInfo, FileInfo> {


// 方法参数的类型为Void,跟我们传入的Void一致,返回类型为 FileInfo ,跟我们传入Result的类型FileInfo一致

@Override

protected FileInfo doInBackground(Void… params) {

}

// 方法参数类型为FileInfo,跟我们传入Progress的类型FileInfo一致

@Override

protected void onProgressUpdate(FileInfo… values) {

}

// 方法参数FileInfo,跟我们传入Result的类型FileInfo一致

@Override

protected void onPostExecute(FileInfo fileInfo) {

}

}

  1. 如果我们更新进度的话,需要重写 onProgressUpdate()方法,并在doInBackground()方法里面调用publishProgress()方法

protected FileInfo doInBackground(Void… params) {

publishProgress(fileInfo);

}

@Override

protected void onProgressUpdate(FileInfo… values) {

super.onProgressUpdate(values);

refreshProgress(values[0]);

}

  1. 当我们调用execute(Params… params) 或者 executeOnExecutor(Executor exec, Params… params) 方法的时候,Task将被防盗相应的 Executor 执行。

MyDownloadTask myDownloadTask = new MyDownloadTask(mDownloadUrl, mDstPath);

myDownloadTask.execute();

完整的Task代码如下

private class MyDownloadTask extends AsyncTask<Void, FileInfo, FileInfo> {

String mDownLoadUrl;

String mDstPath;

public MyDownloadTask(String downloadUrl, String dstPath) {

this.mDownLoadUrl = downloadUrl;

this.mDstPath = dstPath;

}

@Override

protected void onPreExecute() {

super.onPreExecute();

start();

}

@Override

protected FileInfo doInBackground(Void… params) {

//url字符串,检查网址是否已http:// 开头

mDownLoadUrl = (mDownLoadUrl.startsWith(“http://”)) ? mDownLoadUrl : “http://” +

mDownLoadUrl;

Log.d(TAG, “doInBackground: mDownLoadUrl=” + mDownLoadUrl);

Log.d(TAG, “doInBackground: mDstPath=” + mDstPath);

URL url = null;

FileInfo fileInfo = null;

int contentLength = -1;

int downloadLength = 0;

OutputStream output = null;

InputStream istream = null;

try {

url = new URL(mDownLoadUrl);

//打开到url的连接

HttpURLConnection connection = (HttpURLConnection) url.openConnection();

contentLength = connection.getContentLength();

Log.i(TAG, “doInBackground: contentLength=” + contentLength);

//O部分,大体来说就是先检查文件夹是否存在,不存在则创建

istream = connection.getInputStream();

String filename = mDownLoadUrl.substring(mDownLoadUrl.lastIndexOf(“/”) + 1);

File dir = new File(mDstPath);

if (!dir.exists()) {

dir.mkdir();

}

File file = new File(mDstPath + filename);

// 如果存在同名文件,重命名

if (file.exists()) {

file = FileUtils.rename(file.getPath());

}

output = new FileOutputStream(file);

byte[] buffer = new byte[1024 * 4];

int count = 0;

int len = -1;

while ((len = istream.read(buffer)) != -1) {

output.write(buffer, 0, len);

downloadLength += len;

if (count == 10) {

fileInfo = new FileInfo(contentLength, downloadLength, file, file.getPath

(), file.getName());

publishProgress(fileInfo);

count = 0;

}

count++;

}

// 有可能count还没有走到10

fileInfo = new FileInfo(contentLength, downloadLength, file, file.getPath(), file

.getName());

publishProgress(fileInfo);

output.flush();

output.close();

istream.close();

} catch (Exception e) {

e.printStackTrace();

try {

IOUtils.close(output);

IOUtils.close(istream);

} catch (IOException e1) {

e1.printStackTrace();

}

} finally {

try {

IOUtils.close(output);

IOUtils.close(istream);

} catch (IOException e1) {

e1.printStackTrace();

}

}

return fileInfo;

}

@Override

protected void onProgressUpdate(FileInfo… values) {

super.onProgressUpdate(values);

refreshProgress(values[0]);

}

@Override

protected void onPostExecute(FileInfo fileInfo) {

super.onPostExecute(fileInfo);

downloadfinish(fileInfo);

}

@Override

protected void onCancelled() {

super.onCancelled();

}

}

private void start() {

mTvDownloadText.setText(“开始下载”);

mProgressBar.setMax(100);

mProgressBar.setProgress(0);

}

private void downloadfinish(FileInfo fileInfo) {

Log.i(TAG, “onPostExecute: 下载完成=” + fileInfo.mPath);

Toast.makeText(MainActivity.this, “下载完成”, Toast.LENGTH_SHORT).show();

}

private void refreshProgress(FileInfo value) {

FileInfo fileInfo = value;

if (fileInfo != null) {

mProgressBar.setMax((int) fileInfo.mLength);

mProgressBar.setProgress((int) fileInfo.mDownloadLength);

mDownText = fileInfo.mFile.getName() + “下载了” + fileInfo.mDownloadLength + “总长度是” +

fileInfo.mLength;

mTvDownloadText.setText(mDownText);

}

}


AsyncTask的封装使用


前面我们讲完了AsyncTask的基本使用,不知道你有没有发现,其实代码耦合性是挺高的,

- 我们直接在 onProgressUpdata(),onPostExecute()方法里面更新我们的界面,即我们的AsyncTask访问了我们Activity里面的控件,那如果我们修改了Activity的控件,我们岂不是又要去阅读AsyncTask的代码,去做相应的修改。

- 下一次我们如果要下载别的东西,按照我们前面的代码,我们又要重新复制一份,这样无疑是做了很多重复的工作。

说到这样,我相信大多数人的第一感觉就是把AsyncTask提取为外部类,封装起来。是的,确实,我们就是要把AsyncTask提取为外部类。那提取为歪不累之后呢?我们要访问Activity里面的空间,要怎样访问呢?

  1. 在Activity里面定义静态方法

  2. 把需要访问的View对象通过构造函数传递进来

  3. 采用接口回调机制

前面说到的三种方法,是可以做到AsyncTask与外界进行通讯的。但第一第二中方法明显不行。原因如下:

  • 第一种方法定义静态方法,那View对象也必须定义为static变量,这static变量的级别比较高,不易被垃圾回收机制回收,易发生没存泄露。

  • 第二种方法,把需要访问的View对象通过构造函数传递进来。如果需要访问的对象少的话,勉强可以接受,如果多的话,那岂不是要定义很多成员变量。不过最致命的还算是代码耦合性太高了。还不如AsyncTask直接作为内部类。

好了,说了这么多,下面我们一起来看怎样使用接口回调机制来进行解耦。

AsyncTask 使用接口回调机制来进行解耦

  1. 使用接口回调机制,首先我们必须有一个接口

public interface DownloadListener {

void onStart();

void onProgress(FileInfo fileInfo);

void onFinish(FileInfo FileInfo);

void onPaused(FileInfo fileInfo);

void onCancled();

}

  1. 将DownLoadTask提取为一个外部类,并将需要传递的参数传递进来

public class DownloadTask extends AsyncTask<Void,FileInfo,FileInfo> {

private String mDownloadUrl;

private final String mDstPath;

private final String mFileName;

private final DownloadListener mDownloadListener;

public DownloadTask(String downloadUrl, String dstPath, String fileName, DownloadListener downloadListener){

mDownloadUrl = downloadUrl;

mDstPath = dstPath;

mFileName = fileName;

mDownloadListener = downloadListener;

}

}

  1. 在相应的地方调用我们接口的方法

public class DownloadTask extends AsyncTask<Void,FileInfo,FileInfo> {


@Override

protected void onPreExecute() {

super.onPreExecute();

mDownloadListener.onStart();

}

@Override

protected FileInfo doInBackground(Void… params) {


int len = -1;

while ((len = istream.read(buffer)) != -1) {

output.write(buffer, 0, len);

downloadLength += len;

if (count == 10) {

fileInfo = new FileInfo(contentLength, downloadLength, file, file.getPath

(), file.getName());

publishProgress(fileInfo);

count = 0;

}

count++;

}

// 有可能count还没有走到10

fileInfo = new FileInfo(contentLength, downloadLength, file, file.getPath(), file

.getName());

publishProgress(fileInfo);

output.flush();

output.close();

istream.close();

return fileInfo;

}

@Override

protected void onProgressUpdate(FileInfo… values) {

super.onProgressUpdate(values);

mDownloadListener.onProgress(values[0]);

}

尾声

开发是需要一定的基础的,我是08年开始进入Android这行的,在这期间经历了Android的鼎盛时期,和所谓的Android”凉了“。中间当然也有着,不可说的心酸,看着身边朋友,同事一个个转前端,换行业,其实当时我的心也有过犹豫,但是我还是坚持下来了,这次的疫情就是一个好的机会,大浪淘沙,优胜劣汰。再等等,说不定下一个黄金浪潮就被你等到了。

  • 330页 PDF Android核心笔记

  • 几十套阿里 、字节跳动、腾讯、华为、美团等公司2020年的面试题

  • PDF和思维脑图,包含知识脉络 + 诸多细节

  • Android进阶系统学习视频


《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

尾声

开发是需要一定的基础的,我是08年开始进入Android这行的,在这期间经历了Android的鼎盛时期,和所谓的Android”凉了“。中间当然也有着,不可说的心酸,看着身边朋友,同事一个个转前端,换行业,其实当时我的心也有过犹豫,但是我还是坚持下来了,这次的疫情就是一个好的机会,大浪淘沙,优胜劣汰。再等等,说不定下一个黄金浪潮就被你等到了。

  • 330页 PDF Android核心笔记

[外链图片转存中…(img-PWgFwx4J-1714932983253)]

  • 几十套阿里 、字节跳动、腾讯、华为、美团等公司2020年的面试题

[外链图片转存中…(img-v5I3mf9J-1714932983254)]

[外链图片转存中…(img-x6embwk3-1714932983254)]

  • PDF和思维脑图,包含知识脉络 + 诸多细节

[外链图片转存中…(img-KF5OCEnx-1714932983255)]

  • Android进阶系统学习视频

[外链图片转存中…(img-HT4AWAU0-1714932983255)]
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 18
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值