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) {
}
}
- 如果我们更新进度的话,需要重写 onProgressUpdate()方法,并在doInBackground()方法里面调用publishProgress()方法
protected FileInfo doInBackground(Void… params) {
publishProgress(fileInfo);
}
@Override
protected void onProgressUpdate(FileInfo… values) {
super.onProgressUpdate(values);
refreshProgress(values[0]);
}
- 当我们调用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里面的空间,要怎样访问呢?
-
在Activity里面定义静态方法
-
把需要访问的View对象通过构造函数传递进来
-
采用接口回调机制
前面说到的三种方法,是可以做到AsyncTask与外界进行通讯的。但第一第二中方法明显不行。原因如下:
-
第一种方法定义静态方法,那View对象也必须定义为static变量,这static变量的级别比较高,不易被垃圾回收机制回收,易发生没存泄露。
-
第二种方法,把需要访问的View对象通过构造函数传递进来。如果需要访问的对象少的话,勉强可以接受,如果多的话,那岂不是要定义很多成员变量。不过最致命的还算是代码耦合性太高了。还不如AsyncTask直接作为内部类。
好了,说了这么多,下面我们一起来看怎样使用接口回调机制来进行解耦。
AsyncTask 使用接口回调机制来进行解耦
- 使用接口回调机制,首先我们必须有一个接口
public interface DownloadListener {
void onStart();
void onProgress(FileInfo fileInfo);
void onFinish(FileInfo FileInfo);
void onPaused(FileInfo fileInfo);
void onCancled();
}
- 将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;
}
}
- 在相应的地方调用我们接口的方法
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学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!