安卓多线程之异步任务类解决子线程无法返回数据

昨天操作WebAPI的时候由于涉及到主线程(UI线程)与子线程之间的通信时采用Thread创建子线程,并复写 public  void run()方法时,因为run()方法的返回值

类型是void的,所以操作无法把执行完成之后的数据返回给主线程,如果要传递数据到主线程肯定要使用安卓的Handler机制,于是放弃了该方法,并且使用AsyncTask异步任务类

来实现该操作。

下面先来具体说说这个安卓操作线程时必用的异步任务类

Android为了降低这个开发难度,提供了AsyncTask。AsyncTask就是一个封装过的后台任务类,顾名思义就是异步任务。

AsyncTask直接继承于Object类,位置为android.os.AsyncTask。要使用AsyncTask工作我们要提供三个泛型参数,并重载几个方法(至少重载一个)。

 

AsyncTask定义了三种泛型类型 Params,Progress和Result。

  • Params 启动任务执行的输入参数,比如HTTP请求的URL。
  • Progress 后台任务执行的百分比。
  • Result 后台执行任务最终返回的结果,比如String。

使用过AsyncTask 的同学都知道一个异步加载数据最少要重写以下这两个方法:

  • doInBackground(Params…) 后台执行,比较耗时的操作都可以放在这里。注意这里不能直接操作UI。此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用publicProgress(Progress…)来更新任务的进度。
  • onPostExecute(Result)  相当于Handler 处理UI的方式,在这里面可以使用在doInBackground 得到的结果处理操作UI。 此方法在主线程执行,任务执行的结果作为此方法的参数返回

有必要的话你还得重写以下这三个方法,但不是必须的:

  • onProgressUpdate(Progress…)   可以使用进度条增加用户体验度。 此方法在主线程执行,用于显示任务执行的进度。
  • onPreExecute()        这里是最终用户调用Excute时的接口,当任务执行之前开始调用此方法,可以在这里显示进度对话框。
  • onCancelled()             用户调用取消时,要做的操作

使用AsyncTask类,以下是几条必须遵守的准则:

  • Task的实例必须在UI thread中创建;
  • execute方法必须在UI thread中调用;
  • 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法;
  • 该task只能被执行一次,否则多次调用时将会出现异常;

下面是我操作webAPI时的一个简单DEMO,贴出来供大家参考

public class MainActivity extends Activity {

    private String jsonStr="";
    private TextView textView;
    private String TAG="xiedong";

   private String url="your url";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

       textView= (TextView) findViewById(R.id.text);
        MyTask myTask=new MyTask();
        myTask.execute(url);

        Log.i(TAG,jsonStr+"来了");


    }


    /**
     * 为了贴代码,采用了内部类的方式实现
     */
    class MyTask extends AsyncTask<String ,Void,String>{


        /**
         * 执行耗时操作
         * @param params 参数可变
         * @return    返回数据给onPostExecute()使用,工作在UI线程
         *              其他几个方法都是工作在task线程中
         */
        @Override
        protected String doInBackground(String... params) {
            String info="";

            HttpClient client = new DefaultHttpClient();

            HttpGet get = new HttpGet(params[0]);
            try {
               HttpResponse response = client.execute(get);
                if(response.getStatusLine().getStatusCode()==200){

                    InputStream is=response.getEntity().getContent();
                    BufferedReader reader = new BufferedReader(new InputStreamReader(is));

                    String s="";

                    while((s=reader.readLine())!=null){
                        info+=s;
                    }
                }


            } catch (IOException e) {
                e.printStackTrace();
            }
            return info;
        }


        /**
         * 在doInBackground()方法执行之前执行该方法,
         * 完成系统组件的初始化工作
         * 不可手动调用,系统自动调用
         */
        @Override
        protected void onPreExecute() {

        }

        /**
         *
         * @param s  doInBackground()执行完之后传进的参数
         *           在doInBackground()执行完之后执行
         *           把执行的数据返回,不可手动调用,必须是系统自动调用
         */
        @Override
        protected void onPostExecute(String s) {

            //对json数据进行解析
           String jsonStr = getJsonStr(s);
            Log.i(TAG,jsonStr+"json");
            textView.setText(jsonStr);

        }
    }

    private String getJsonStr(String s) {

        String data="";
        try {
            //实例化json数组
//            JSONArray array =new JSONArray(s);

            //得到第一个节点的json字符串
//            JSONObject obj = array.getJSONObject(0);

            JSONObject obj =new JSONObject(s);
            data=obj.getString("info");
            Log.i(TAG,data);

        } catch (JSONException e) {
            e.printStackTrace();
        }

        return data;
    }

}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值