做android开发的都知道,我们在主线程中不能进行耗时太久的操作,否则就可能报ANR,所有耗时的操作都要异步进行。android中的异步方法 也很多,可以用Handler,可以用Thread及Runnable等。今天我们用安卓自带的异步工具AsyncTask来实现耗时操作:从网络下载文 件。先对AsyncTask作个简单的介绍,下面的介绍也主要是参考网上资料:
1,AsyncTask定义了三种泛型类型 Params,Progress和Result。
- Params 启动任务执行的输入参数,如网络请求的URL。
- Progress 后台进行中任务执行进度百分比。
- Result 后台执行任务完成最终返回的结果,比如String,Bitmap等。
2,使用AsyncTask 最少要重写以下这两个方法:
- doInBackground(Params…) 后台执行,耗时的操作都可以放在这里(这里不能直接操作UI)。在执行过程中可以调用publicProgress(Progress…)来更新任务的进度,并用进度条显示出来。
- onPostExecute(Result) 相当于Handler 处理UI的方式,在这里面可以使用在doInBackground 得到的结果处理操作UI。 此方法在主线程执行,任务执行的结果作为此方法的参数返回。
3,有进我还需要重写以下这三个方法(非必需):
- onProgressUpdate(Progress…) 可以使用进度条增加用户体验度。 此方法在主线程执行,用于显示任务执行的进度。
- onPreExecute() 当任务执行之前开始调用此方法,可以在这里显示进度对话框。
- onCancelled() 用户调用取消时,要做的操作可以放到这里面处理。
4,使用AsyncTask类,以下是几条必须遵守的准则:
- AsyncTask的实例和execute方法必须在UI线程中创建,
- 不要手动调用onPreExecute(), onPostExecute(Result),doInBackground(Params...),onProgressUpdate(Progress...)这4个方法,
- 该实现的AsyncTask只能被执行一次,否则多次调用时将会出现异常; 下面是我们实现文件下载的例子:
- 布局:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <Button android:id="@+id/file_download_btn" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:textSize="16sp" android:text="下载文件" /> </LinearLayout>
- Activity代码:
public class MainActivity extends Activity implements OnClickListener { /* SD卡根目录 */ private File rootDie; /* 输出文件名称 */ private String outFileName = "ldm.rar"; /* 进度条对话框 */ private ProgressDialog pdialog; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); checkAndCreateDir(); findViewById(R.id.file_download_btn).setOnClickListener(this); } @Override protected Dialog onCreateDialog(int id) { /* 实例化进度条对话框 */ pdialog = new ProgressDialog(this); /* 进度条对话框属性设置 */ pdialog.setMessage("正在下载中..."); /* 进度值最大100 */ pdialog.setMax(100); /* 水平风格进度条 */ pdialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL); /* 无限循环模式 */ pdialog.setIndeterminate(false); /* 可取消 */ pdialog.setCancelable(true); /* 显示对话框 */ pdialog.show(); return pdialog; } /* 检查sdcard并创建目录文件 */ private void checkAndCreateDir() { /* 获取sdcard目录 */ rootDie = Environment.getExternalStorageDirectory(); /* 新文件的目录 */ File newFile = new File(rootDie + "/download/"); if (!newFile.exists()) { /* 如果文件不存在就创建目录 */ newFile.mkdirs(); } } /* 异步任务,后台处理与更新UI */ class MyLoadAsyncTask extends AsyncTask<String, String, String> { /* 后台线程 */ @Override protected String doInBackground(String... params) { /* 所下载文件的URL */ try { URL url = new URL(params[0]); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); /* URL属性设置 */ conn.setRequestMethod("GET"); /* URL建立连接 */ conn.connect(); /* 下载文件的大小 */ int fileOfLength = conn.getContentLength(); /* 每次下载的大小与总下载的大小 */ int totallength = 0; int length = 0; /* 输入流 */ InputStream in = conn.getInputStream(); /* 输出流 */ FileOutputStream out = new FileOutputStream(new File(rootDie + "/mydownload1/", outFileName)); /* 缓存模式,下载文件 */ byte[] buff = new byte[1024 * 1024]; while ((length = in.read(buff)) > 0) { totallength += length; String str1 = "" + (int) ((totallength * 100) / fileOfLength); publishProgress(str1); out.write(buff, 0, length); } /* 关闭输入输出流 */ in.close(); out.flush(); out.close(); } catch (MalformedURLException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } return null; } /* 预处理UI线程 */ @Override protected void onPreExecute() { showDialog(0); super.onPreExecute(); } /* 结束时的UI线程 */ @Override protected void onPostExecute(String result) { dismissDialog(0); super.onPostExecute(result); } /* 处理UI线程,会被多次调用,触发事件为publicProgress方法 */ @Override protected void onProgressUpdate(String... values) { /* 进度显示 */ pdialog.setProgress(Integer.parseInt(values[0])); } } @Override public void onClick(View v) { if (v.getId() == R.id.file_download_btn) { /* 异步下载 */ new MyLoadAsyncTask().execute("http://192.168.1.105:8080/tool.rar"); } } }
- 最后别忘记在AndroidManifest.xml中加上权限:
<!-- sdcard读写权限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <!-- sdcard创建目录与文件权限 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" /> <!-- 网络访问权限 --> <uses-permission android:name="android.permission.INTERNET" />