当前位置:  编程技术>移动开发
本页文章导读:
    ▪GridView和边距有关问题        GridView和边距问题 今天在弄GridView的时候碰到GridView里面的item和外边框之间存在一定的距离,整了半天,也搜索过,还是没咋找到原因,抱着试试看的心态把它的android:listSelector值设置成了.........
    ▪ 查询包孕某个字段难道所有表名        查询包含某个字段难道所有表名 SELECT c.name,o.name FROM syscolumns AS c INNER JOIN sysobjects AS o ON c.id = o.id where c.name='ServicePackageID' ......
    ▪ AsyncTask的兑现原理       AsyncTask的实现原理 转自: http://blog.csdn.net/mylzc/article/details/6774131   在《Android异步处理二:使用AsyncTask异步更新UI界面》一文中,我们介绍了如何使用AsyncTask实现异步下载图片,并且更新图.........

[1]GridView和边距有关问题
    来源: 互联网  发布时间: 2014-02-18
GridView和边距问题
今天在弄GridView的时候碰到GridView里面的item和外边框之间存在一定的距离,整了半天,也搜索过,还是没咋找到原因,抱着试试看的心态把它的
android:listSelector值设置成了 @null,问题竟然就这样解决了,汗死,搞半天原来就这个原因。。。
无语中。。。

    
[2] 查询包孕某个字段难道所有表名
    来源: 互联网  发布时间: 2014-02-18
查询包含某个字段难道所有表名
SELECT c.name,o.name FROM syscolumns AS c
INNER JOIN sysobjects AS o
ON c.id = o.id where c.name='ServicePackageID'



    
[3] AsyncTask的兑现原理
    来源: 互联网  发布时间: 2014-02-18
AsyncTask的实现原理

转自: http://blog.csdn.net/mylzc/article/details/6774131

 

在《Android异步处理二:使用AsyncTask异步更新UI界面》一文中,我们介绍了如何使用AsyncTask实现异步下载图片,并且更新图片到UI界面的方法。本篇我们将学习Framework层AsyncTask的实现原理。

概述:AsyncTask的本质是一个线程池,所有提交的异步任务都会在这个线程池中的工作线程内执行,当工作线程需要跟UI线程交互时,工作线程会通过向在UI线程创建的Handler(原理见:《Android异步处理三:Handler+Looper+MessageQueue深入详解》)传递消息的方式,调用相关的回调函数,从而实现UI界面的更新。

 

例子:

本章还是以《Android异步处理二:使用AsyncTask异步更新UI界面》中的例子说明AsyncTask的实现原理。

这个例子是在后台下载CSDN的LOGO,下载完成后在UI界面上显示出来。

 

AsyncTask.java

 

[java] view plaincopy
 
  • package com.zhuozhuo;  
  •   
  •   
  • import org.apache.http.HttpResponse;  
  • import org.apache.http.client.HttpClient;  
  • import org.apache.http.client.methods.HttpGet;  
  • import org.apache.http.impl.client.DefaultHttpClient;  
  •   
  • import android.app.Activity;  
  • import android.graphics.Bitmap;  
  • import android.graphics.BitmapFactory;  
  • import android.os.AsyncTask;  
  • import android.os.Bundle;  
  • import android.view.View;  
  • import android.view.View.OnClickListener;  
  • import android.widget.Button;  
  • import android.widget.ImageView;  
  • import android.widget.ProgressBar;  
  • import android.widget.Toast;  
  •   
  • public class AsyncTaskActivity extends Activity {  
  •       
  •     private ImageView mImageView;  
  •     private Button mButton;  
  •     private ProgressBar mProgressBar;  
  •       
  •     @Override  
  •     public void onCreate(Bundle savedInstanceState) {  
  •         super.onCreate(savedInstanceState);  
  •         setContentView(R.layout.main);  
  •           
  •         mImageView= (ImageView) findViewById(R.id.imageView);  
  •         mButton = (Button) findViewById(R.id.button);  
  •         mProgressBar = (ProgressBar) findViewById(R.id.progressBar);  
  •         mButton.setOnClickListener(new OnClickListener() {  
  •               
  •             @Override  
  •             public void onClick(View v) {  
  •                 GetCSDNLogoTask task = new GetCSDNLogoTask();  
  •                 task.execute("http://csdnimg.cn/www/images/csdnindex_logo.gif");  
  •             }  
  •         });  
  •     }  
  •       
  •     class GetCSDNLogoTask extends AsyncTask<String,Integer,Bitmap> {//继承AsyncTask  
  •   
  •         @Override  
  •         protected Bitmap doInBackground(String... params) {//处理后台执行的任务,在后台线程执行  
  •             publishProgress(0);//将会调用onProgressUpdate(Integer... progress)方法  
  •             HttpClient hc = new DefaultHttpClient();  
  •             publishProgress(30);  
  •             HttpGet hg = new HttpGet(params[0]);//获取csdn的logo  
  •             final Bitmap bm;  
  •             try {  
  •                 HttpResponse hr = hc.execute(hg);  
  •                 bm = BitmapFactory.decodeStream(hr.getEntity().getContent());  
  •             } catch (Exception e) {  
  •                   
  •                 return null;  
  •             }  
  •             publishProgress(100);  
  •             //mImageView.setImageBitmap(result); 不能在后台线程操作ui  
  •             return bm;  
  •         }  
  •           
  •         protected void onProgressUpdate(Integer... progress) {//在调用publishProgress之后被调用,在ui线程执行  
  •             mProgressBar.setProgress(progress[0]);//更新进度条的进度  
  •          }  
  •   
  •          protected void onPostExecute(Bitmap result) {//后台任务执行完之后被调用,在ui线程执行  
  •              if(result != null) {  
  •                  Toast.makeText(AsyncTaskActivity.this, "成功获取图片", Toast.LENGTH_LONG).show();  
  •                  mImageView.setImageBitmap(result);  
  •              }else {  
  •                  Toast.makeText(AsyncTaskActivity.this, "获取图片失败", Toast.LENGTH_LONG).show();  
  •              }  
  •          }  
  •            
  •          protected void onPreExecute () {//在 doInBackground(Params...)之前被调用,在ui线程执行  
  •              mImageView.setImageBitmap(null);  
  •              mProgressBar.setProgress(0);//进度条复位  
  •          }  
  •            
  •          protected void onCancelled () {//在ui线程执行  
  •              mProgressBar.setProgress(0);//进度条复位  
  •          }  
  •           
  •     }  
  •       
  •   
  • }  
  • main.xml

     

     

    [html] view plaincopy
     
  • <?xml version="1.0" encoding="utf-8"?>  
  • <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  •     android:orientation="vertical" android:layout_width="fill_parent"  
  •     android:layout_height="fill_parent">  
  •     <ProgressBar android:layout_width="fill_parent" android:layout_height="wrap_content" android:id="@+id/progressBar" ></ProgressBar>  
  •     <Button android:id="@+id/button" android:text="下载图片" android:layout_width="wrap_content" android:layout_height="wrap_content"></Button>  
  •     <ImageView android:id="@+id/imageView" android:layout_height="wrap_content"  
  •         android:layout_width="wrap_content" />  
  • </LinearLayout>  
  • mainifest.xml

    [html] view plaincopy
     
  • <?xml version="1.0" encoding="utf-8"?>  
  • <manifest xmlns:android="http://schemas.android.com/apk/res/android"  
  •       package="com.zhuozhuo"  
  •       android:versionCode="1"  
  •       android:versionName="1.0">  
  •     <uses-sdk android:minSdkVersion="10" />  
  • <uses-permission android:name="android.permission.INTERNET"></uses-permission>  
  •     <application android:icon="@drawable/icon" android:label="@string/app_name">  
  •         <activity android:name=".AsyncTaskActivity"  
  •                   android:label="@string/app_name">  
  •             <intent-filter>  
  •                 <action android:name="android.intent.action.MAIN" />  
  •                 <category android:name="android.intent.category.LAUNCHER" />  
  •             </intent-filter>  
  •         </activity>  
  •   
  •     </application>  
  • </manifest>  
  •  

    运行结果:

     



     

    分析:

    在分析实现流程之前,我们先了解一下AsyncTask有哪些成员变量。

     

    [java] view plaincopy
     
  • private static final int CORE_POOL_SIZE =5;//5个核心工作线程  
  • private static final int MAXIMUM_POOL_SIZE = 128;//最多128个工作线程  
  • private static final int KEEP_ALIVE = 1;//空闲线程的超时时间为1秒  
  •   
  • private static final BlockingQueue<Runnable> sWorkQueue =  
  •         new LinkedBlockingQueue<Runnable>(10);//等待队列  
  •   
  • private static final ThreadPoolExecutorsExecutor = new ThreadPoolExecutor(CORE_POOL_SIZE,  
  •         MAXIMUM_POOL_SIZE, KEEP_ALIVE, TimeUnit.SECONDS, sWorkQueue,sThreadFactory);//线程池是静态变量,所有的异步任务都会放到这个线程池的工作线程内执行。  


  •  

     

    回到例子中,点击按钮之后会新建一个GetCSDNLogoTask对象:

     

    [java] view plaincopy
     
  • GetCSDNLogoTask task = new GetCSDNLogoTask();  
  •  

     

    此时会调用父类AsyncTask的构造函数:

    AsyncTask.java

     

    [java] view plaincopy
     
  • public AsyncTask() {  
  •         mWorker = new WorkerRunnable<Params, Result>() {  
  •             public Result call() throws Exception {  
  •                 Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);  
  •                 return doInBackground(mParams);  
  •             }  
  •         };  
  •   
  •         mFuture = new FutureTask<Result>(mWorker) {  
  •             @Override  
  •             protected void done() {  
  •                 Message message;  
  •                 Result result = null;  
  •   
  •                 try {  
  •                     result = get();  
  •                 } catch (InterruptedException e) {  
  •                     android.util.Log.w(LOG_TAG, e);  
  •                 } catch (ExecutionException e) {  
  •                     throw new RuntimeException("An error occured while executing doInBackground()",  
  •                             e.getCause());  
  •                 } catch (CancellationException e) {  
  •                     message = sHandler.obtainMessage(MESSAGE_POST_CANCEL,  
  •                             new AsyncTaskResult<Result>(AsyncTask.this, (Result[]) null));  
  •                     message.sendToTarget();//取消任务,发送MESSAGE_POST_CANCEL消息  
  •                     return;  
  •                 } catch (Throwable t) {  
  •                     throw new RuntimeException("An error occured while executing "  
  •                             + "doInBackground()", t);  
  •                 }  
  •   
  •                 message = sHandler.obtainMessage(MESSAGE_POST_RESULT,  
  •                         new AsyncTaskResult<Result>(AsyncTask.this, result));//完成任务,发送MESSAGE_POST_RESULT消息并传递result对象  
  •                 message.sendToTarget();  
  •             }  
  •         };  
  •     }  


  •  

    WorkerRunnable类实现了callable接口的call()方法,该函数会调用我们在AsyncTask子类中实现的doInBackground(mParams)方法,由此可见,WorkerRunnable封装了我们要执行的异步任务。FutureTask中的protected void done() {}方法实现了异步任务状态改变后的操作。当异步任务被取消,会向UI线程传递MESSAGE_POST_CANCEL消息,当任务成功执行,会向UI线程传递MESSAGE_POST_RESULT消息,并把执行结果传递到UI线程。

    由此可知,AsyncTask在构造的时候已经定义好要异步执行的方法doInBackground(mParams)和任务状态变化后的操作(包括失败和成功)。

    当创建完GetCSDNLogoTask对象后,执行

     

    [java] view plaincopy
     
  • task.execute("http://csdnimg.cn/www/images/csdnindex_logo.gif");  
  •  

     

    此时会调用AsyncTask的execute(Params...params)方法

     

    AsyncTask.java

     

    [java] view plaincopy
     
  • public final AsyncTask<Params,Progress, Result> execute(Params... params) {  
  •         if (mStatus != Status.PENDING) {  
  •             switch (mStatus) {  
  •                 case RUNNING:  
  •                     throw newIllegalStateException("Cannot execute task:"  
  •                             + " the taskis already running.");  
  •                 case FINISHED:  
  •                     throw newIllegalStateException("Cannot execute task:"  
  •                             + " the taskhas already been executed "  
  •                             + "(a task canbe executed only once)");  
  •             }  
  •         }  
  •    
  •         mStatus = Status.RUNNING;  
  •    
  •         onPreExecute();//运行在ui线程,在提交任务到线程池之前执行  
  •    
  •         mWorker.mParams = params;  
  •         sExecutor.execute(mFuture);//提交任务到线程池  
  •    
  •         return this;  
  • }  


  •  

    当任务正在执行或者已经完成,会抛出IllegalStateException,由此可知我们不能够重复调用execute(Params...params)方法。在提交任务到线程池之前,调用了onPreExecute()方法。然后才执行sExecutor.execute(mFuture)是任务提交到线程池。

    前面我们说到,当任务的状态发生改变时(1、执行成功2、取消执行3、进度更新),工作线程会向UI线程的Handler传递消息。在《Android异步处理三:Handler+Looper+MessageQueue深入详解》一文中我们提到,Handler要处理其他线程传递过来的消息。在AsyncTask中,InternalHandler是在UI线程上创建的,它接收来自工作线程的消息,实现代码如下:

     

    AsyncTask.java

     

    [java] view plaincopy
     
  • private static class InternalHandler extends Handler {  
  •        @SuppressWarnings({"unchecked","RawUseOfParameterizedType"})  
  •         @Override  
  •         public voidhandleMessage(Message msg) {  
  •             AsyncTaskResult result =(AsyncTaskResult) msg.obj;  
  •             switch (msg.what) {  
  •                 caseMESSAGE_POST_RESULT:  
  •                     // There is onlyone result  
  •                     result.mTask.finish(result.mData[0]);//执行任务成功  
  •                     break;  
  •                 caseMESSAGE_POST_PROGRESS:  
  •                    result.mTask.onProgressUpdate(result.mData);//进度更新  
  •                     break;  
  •                 caseMESSAGE_POST_CANCEL:  
  •                     result.mTask.onCancelled();//取消任务  
  •                     break;  
  •             }  
  •         }  
  •     }  


  •  

    当接收到消息之后,AsyncTask会调用自身相应的回调方法。

    总结:

    1、 AsyncTask的本质是一个静态的线程池,AsyncTask派生出的子类可以实现不同的异步任务,这些任务都是提交到静态的线程池中执行。

    2、线程池中的工作线程执行doInBackground(mParams)方法执行异步任务

     

    3、当任务状态改变之后,工作线程会向UI线程发送消息,AsyncTask内部的InternalHandler响应这些消息,并调用相关的回调函数


        
    最新技术文章:
    ▪Android开发之登录验证实例教程
    ▪Android开发之注册登录方法示例
    ▪Android获取手机SIM卡运营商信息的方法
    ▪Android实现将已发送的短信写入短信数据库的...
    ▪Android发送短信功能代码
    ▪Android根据电话号码获得联系人头像实例代码
    ▪Android中GPS定位的用法实例
    ▪Android实现退出时关闭所有Activity的方法
    ▪Android实现文件的分割和组装
    ▪Android录音应用实例教程
    ▪Android双击返回键退出程序的实现方法
    ▪Android实现侦听电池状态显示、电量及充电动...
    ▪Android获取当前已连接的wifi信号强度的方法
    ▪Android实现动态显示或隐藏密码输入框的内容
    ▪根据USER-AGENT判断手机类型并跳转到相应的app...
    ▪Android Touch事件分发过程详解
    ▪Android中实现为TextView添加多个可点击的文本
    ▪Android程序设计之AIDL实例详解
    ▪Android显式启动与隐式启动Activity的区别介绍
    ▪Android按钮单击事件的四种常用写法总结
    ▪Android消息处理机制Looper和Handler详解
    ▪Android实现Back功能代码片段总结
    ▪Android实用的代码片段 常用代码总结
    ▪Android实现弹出键盘的方法
    ▪Android中通过view方式获取当前Activity的屏幕截...
    ▪Android提高之自定义Menu(TabMenu)实现方法
    ▪Android提高之多方向抽屉实现方法
    ▪Android提高之MediaPlayer播放网络音频的实现方法...
    ▪Android提高之MediaPlayer播放网络视频的实现方法...
    ▪Android提高之手游转电视游戏的模拟操控
     


    站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3