在开发Android应用时必须遵守单线程模型的原则: Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行。在单线程模型中始终要记住两条法则:
1. 不要阻塞UI线程
2. 确保只在UI线程中访问Android UI工具包
当一个程序第一次启动时,Android会同时启动一个对应的主线程(Main Thread),主线程主要负责处理与UI相关的事件,如:用户的按键事件,用户接触屏幕的事件以及屏幕绘图事件,并把相关的事件分发到对应的组件进行处理。所以主线程通常又被叫做UI线程。
比如说从网上获取一个网页,在一个TextView中将其源代码显示出来,这种涉及到网络操作的程序一般都是需要开一个线程完成网络访问,但是在获得页面源码后,是不能直接在网络操作线程中调用TextView.setText()的.因为其他线程中是不能直接访问主UI线程成员


android提供了几种在其他线程中访问UI线程的方法。
Activity.runOnUiThread( Runnable )
View.post( Runnable )
View.postDelayed( Runnable, long )
Hanlder
这些类或方法同样会使你的代码很复杂很难理解。然而当你需要实现一些很复杂的操作并需要频繁地更新UI时这会变得更糟糕。

为了解决这个问题,Android 1.5提供了一个工具类:AsyncTask,它使创建需要与用户界面交互的长时间运行的任务变得更简单。不需要借助线程和Handler即可实现。
AsyncTask是抽象类.AsyncTask定义了三种泛型类型 Params,Progress和Result。
  Params 启动任务执行的输入参数,比如HTTP请求的URL。
  Progress 后台任务执行的百分比。
  Result 后台执行任务最终返回的结果,比如String。

AsyncTask的执行分为四个步骤,每一步都对应一个回调方法,这些方法不应该由应用程序调用,开发者需要做的就是实现这些方法。
  1) 子类化AsyncTask
  2) 实现AsyncTask中定义的下面一个或几个方法
   onPreExecute(), 该方法将在执行实际的后台操作前被UI thread调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条。
   doInBackground(Params...), 将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些很耗时的后台计算工作。可以调用 publishProgress方法来更新实时的任务进度。该方法是抽象方法,子类必须实现。
   onProgressUpdate(Progress...),在publishProgress方法被调用后,UI thread将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。
   onPostExecute(Result), 在doInBackground 执行完成后,onPostExecute 方法将被UI thread调用,后台的计算结果将通过该方法传递到UI thread.

为了正确的使用AsyncTask类,以下是几条必须遵守的准则:
  1) Task的实例必须在UI thread中创建
  2) execute方法必须在UI thread中调用
  3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法
  4) 该task只能被执行一次,否则多次调用时将会出现异常


从网上获取一个网页,在一个TextView中将其源代码显示出来

Java代码
  1. packagetest.list;
  2. importjava.io.ByteArrayOutputStream;
  3. importjava.io.InputStream;
  4. importjava.util.ArrayList;
  5. importorg.apache.http.HttpEntity;
  6. importorg.apache.http.HttpResponse;
  7. importorg.apache.http.client.HttpClient;
  8. importorg.apache.http.client.methods.HttpGet;
  9. importorg.apache.http.impl.client.DefaultHttpClient;
  10. importandroid.app.Activity;
  11. importandroid.app.ProgressDialog;
  12. importandroid.content.Context;
  13. importandroid.content.DialogInterface;
  14. importandroid.os.AsyncTask;
  15. importandroid.os.Bundle;
  16. importandroid.os.Handler;
  17. importandroid.os.Message;
  18. importandroid.view.View;
  19. importandroid.widget.Button;
  20. importandroid.widget.EditText;
  21. importandroid.widget.TextView;
  22. publicclassNetworkActivityextendsActivity{
  23. privateTextViewmessage;
  24. privateButtonopen;
  25. privateEditTexturl;
  26. @Override
  27. publicvoidonCreate(BundlesavedInstanceState){
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.network);
  30. message=(TextView)findViewById(R.id.message);
  31. url=(EditText)findViewById(R.id.url);
  32. open=(Button)findViewById(R.id.open);
  33. open.setOnClickListener(newView.OnClickListener(){
  34. publicvoidonClick(Viewarg0){
  35. connect();
  36. }
  37. });
  38. }
  39. privatevoidconnect(){
  40. PageTasktask=newPageTask(this);
  41. task.execute(url.getText().toString());
  42. }
  43. classPageTaskextendsAsyncTask<String,Integer,String>{
  44. //可变长的输入参数,与AsyncTask.exucute()对应
  45. ProgressDialogpdialog;
  46. publicPageTask(Contextcontext){
  47. pdialog=newProgressDialog(context,0);
  48. pdialog.setButton("cancel",newDialogInterface.OnClickListener(){
  49. publicvoidonClick(DialogInterfacedialog,inti){
  50. dialog.cancel();
  51. }
  52. });
  53. pdialog.setOnCancelListener(newDialogInterface.OnCancelListener(){
  54. publicvoidonCancel(DialogInterfacedialog){
  55. finish();
  56. }
  57. });
  58. pdialog.setCancelable(true);
  59. pdialog.setMax(100);
  60. pdialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
  61. pdialog.show();
  62. }
  63. @Override
  64. protectedStringdoInBackground(String...params){
  65. try{
  66. HttpClientclient=newDefaultHttpClient();
  67. //params[0]代表连接的url
  68. HttpGetget=newHttpGet(params[0]);
  69. HttpResponseresponse=client.execute(get);
  70. HttpEntityentity=response.getEntity();
  71. longlength=entity.getContentLength();
  72. InputStreamis=entity.getContent();
  73. Strings=null;
  74. if(is!=null){
  75. ByteArrayOutputStreambaos=newByteArrayOutputStream();
  76. byte[]buf=newbyte[128];
  77. intch=-1;
  78. intcount=0;
  79. while((ch=is.read(buf))!=-1){
  80. baos.write(buf,0,ch);
  81. count+=ch;
  82. if(length>0){
  83. //如果知道响应的长度,调用publishProgress()更新进度
  84. publishProgress((int)((count/(float)length)*100));
  85. }
  86. //让线程休眠100ms
  87. Thread.sleep(100);
  88. }
  89. s=newString(baos.toByteArray());}
  90. //返回结果
  91. returns;
  92. }catch(Exceptione){
  93. e.printStackTrace();
  94. }
  95. returnnull;
  96. }
  97. @Override
  98. protectedvoidonCancelled(){
  99. super.onCancelled();
  100. }
  101. @Override
  102. protectedvoidonPostExecute(Stringresult){
  103. //返回HTML页面的内容
  104. message.setText(result);
  105. pdialog.dismiss();
  106. }
  107. @Override
  108. protectedvoidonPreExecute(){
  109. //任务启动,可以在这里显示一个对话框,这里简单处理
  110. message.setText(R.string.task_started);
  111. }
  112. @Override
  113. protectedvoidonProgressUpdate(Integer...values){
  114. //更新进度
  115. System.out.println(""+values[0]);
  116. message.setText(""+values[0]);
  117. pdialog.setProgress(values[0]);
  118. }
  119. }
  120. }

更多相关文章

  1. 教你如何在 Android 使用多线程下载文件
  2. Android设置TextView显示指定个数字符,超过部分显示...(省略号)的
  3. Android Service——在子线程中更新UI
  4. Android使用自定义字体的方法
  5. Android 环信官方Demo3.3.2详细配置方法
  6. Android GLES多线程处理
  7. Android中经常用到的方法--SDcard下文件的操作
  8. Android Intent调用方法总结
  9. Android 应用程序退出后不在运行列表中显示的方法

随机推荐

  1. Android中的Adapter 详解(三)
  2. android 工程师 你必须知道的
  3. Android H5和App交互以及打开图库上传图
  4. 通读Android文档系列 ONE
  5. 一篇文章带你了解 Android 消息机制的原
  6. Android之drawable和mipmap目录区别
  7. 【多图】Google工程师解析Android系统架
  8. Android(安卓)Glide图片加载框架介绍
  9. Android屏幕适配总结
  10. Android 集成支付宝支付详解