AsyncTask (API level 3,所以几乎所有目前在市面上流通的 AndroID 版本皆可使用)
是除 Thread 外的另一种选择,AndroID 团队鼓励主执行绪(UI thread) 专注于 *** 作 & 画面的流畅呈现,
其余工作 (如网络资料传输、档案/磁碟/资料存取) 最好都在背景执行;
Thread 通常要搭配 Handler 使用,而 AsyncTask 用意在简化背景执行 thread 程序码的撰写。
如果您预期要执行的工作能在几秒内完成,就可以选择使用 AsyncTask,若执行的时间很长,
AndroID 则强烈建议采用 Executor,ThreadPoolExecutor and FutureTask。
要使用 AsyncTask,必定要建立一个继承自 AsyncTask 的子类别,并传入 3 项资料:
Params -- 要执行 doInBackground() 时传入的参数,数量可以不止一个 Progress -- doInBackground() 执行过程中回传给 UI thread 的资料,数量可以不止一个 Rsesult -- 传回执行结果若您没有参数要传入,则填入 VoID (注意 V 为大写)。
AsyncTask 的运作有 4 个阶段:
onPreExecute -- AsyncTask 执行前的准备工作,例如画面上显示进度表, doInBackground -- 实际要执行的程序码就是写在这里, onProgressUpdate -- 用来显示目前的进度, onPostExecute -- 执行完的结果 - Result 会传入这里。除了 doInBackground,其他 3 个 method 都是在 UI thread 呼叫
炫酷进度条实例
我们以一个实例来说明,“点击按钮开始下载QQAndroID安装包,然后显示一个对话框来反馈下载进度”。我们先初始化一个对话框,由于要显示进度,我们用Github上面一个能够显示百分比的进度条 NumberProgressbar,启动任务的按钮我们使用 circlebutton,一个有酷炫动画的按钮,Github上面有很多非常好的开源项目,当然炫酷的控件是其中一部分了,后面有机会,会去学习一些比较流行的控件它们的实现原理,今天就暂且拿来主义了~~。
1.先初始化进度条提示对话框。
builder = new AlertDialog.Builder( MainActivity.this); LayoutInflater inflater = LayoutInflater.from(MainActivity.this); mDialogVIEw = inflater.inflate(R.layout.progress_dialog_layout,null); mNumberProgressbar = (NumberProgressbar)mDialogVIEw.findVIEwByID(R.ID.number_progress_bar); builder.setVIEw(mDialogVIEw); mDialog = builder.create();
2.设置按钮点击事件。
findVIEwByID(R.ID.circle_btn).setonClickListener(new VIEw.OnClickListener(){ @OverrIDe public voID onClick(VIEw v) { dismissDialog(); mNumberProgressbar.setProgress(0); myTask = new MyAsyncTask(); myTask.execute(qqDownloadUrl); } });
3.DownloadAsyncTask实现,有点长。
private class DownloadAsyncTask extends AsyncTask<String,Integer,String> { @OverrIDe protected voID onPreExecute() { super.onPreExecute(); mDialog.show(); } @OverrIDe protected voID onPostExecute(String aVoID) { super.onPostExecute(aVoID); dismissDialog(); } @OverrIDe protected voID onProgressUpdate(Integer... values) { super.onProgressUpdate(values); mNumberProgressbar.setProgress(values[0]); } @OverrIDe protected voID onCancelled(String aVoID) { super.onCancelled(aVoID); dismissDialog(); } @OverrIDe protected voID onCancelled() { super.onCancelled(); dismissDialog(); } @OverrIDe protected String doInBackground(String... params) { String urlStr = params[0]; fileOutputStream output = null; try { URL url = new URL(urlStr); httpURLConnection connection = (httpURLConnection)url.openConnection(); String qqApkfile = "qqApkfile"; file file = new file(Environment.getExternalStorageDirectory() + "/" + qqApkfile); if (file.exists()) { file.delete(); } file.createNewfile(); inputStream input = connection.getinputStream(); output = new fileOutputStream(file); int total = connection.getContentLength(); if (total <= 0) { return null; } int plus = 0; int totalRead = 0; byte[] buffer = new byte[4*1024]; while((plus = input.read(buffer)) != -1){ output.write(buffer); totalRead += plus; publishProgress(totalRead * 100 / total); if (isCancelled()) { break; } } output.flush(); } catch (MalformedURLException e) { e.printstacktrace(); if (output != null) { try { output.close(); } catch (IOException e2) { e2.printstacktrace(); } } } catch (IOException e) { e.printstacktrace(); if (output != null) { try { output.close(); } catch (IOException e2) { e2.printstacktrace(); } } } finally { if (output != null) { try { output.close(); } catch (IOException e) { e.printstacktrace(); } } } return null; }}
这样一个简单的下载文件文件就基本实现了,到目前为止谈不上技巧,但是现在我们有一个问题,就是如果我们的Activity正在后台执行一个任务,可能耗时较长,那用户可能会点击返回退出Activity或者退出App,那么后台任务不会立即退出,如果AsyncTask内部有Activity中成员变量的引用,还会造成Activity的回收延时,造成一段时间内的内存泄露,所以我们需要加上下面的第四步处理。
4.onPause中判断应用是否要退出,从而决定是否取消AsyncTask执行。
@OverrIDeprotected voID onPause() { super.onPause(); if (myTask != null && isFinishing()) { myTask.cancel(false); }}
这样我们的异步任务就会在Activity退出时,也随之取消任务执行,顺利被系统销毁回收,第四步很多时候会被遗漏,而且一般也不会有什么致命的问题,但是一旦出问题了,就很难排查,所以遵循编码规范还是有必要的。
以上是内存溢出为你收集整理的Android中通过AsyncTask类来制作炫酷进度条的实例教程全部内容,希望文章能够帮你解决Android中通过AsyncTask类来制作炫酷进度条的实例教程所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)