Android多线程的简要分析

来源:互联网 发布:mac磁盘清理工具 编辑:程序博客网 时间:2024/04/24 07:09
 

在J2ME中,多线程的处理是很简单的,似乎并不用考虑很多问题,顶多是同步方面很让人头疼,而在Android中,一切似乎不那么明了了,很重要的一点是在其他线程中是不能直接访问主UI线程成员的。比如说从网上获取一个网页,在一个TextView中将其源代码显示出来,这种涉及到网络操作的程序一般都是需要开一个线程完成网络访问,但是在获得页面源码后,是不能直接在网络操作线程中调用TextView.setText()的,在C#中也是这样,可能是为了主线程的安全吧。
对于此类情况,Android提供了一些间接访问的方法。

1.runOnUiThread函数
这个函数是Activity的成员函数,格式是public final void runOnUiThread (Runnable action)
在自定义线程中执行对应Activity的runOnUiThread函数就好了,但是很明显,参数还是个线程,这个

函数基本上没什么意义。

2.Handler
这应该是Android多线程中用的最多的工具了,这里Android借鉴了Windows的消息机制,不同线程之间可以用消息进行交流。除了主线程(UI线程)外,通过继承Thread或Runnable生成的线程是不包含消息循环(Looper)的。因此必须在线程中调用Looper.prepare()准备一个消息循环,调用Looper.loop()开始循环,就像是这样:
class LooperThread extends Thread {
      public Handler mHandler;
      public void run() {
          Looper.prepare();
          mHandler = new Handler() {
              public void handleMessage(Message msg) {
                  // process incoming messages here
              }
          };
          Looper.loop();
      }
}
这其中Handler是用来接受并处理消息的,如果像上面写的那样调用无参数构造函数生成handler,这个Handler对应的就是所在线程,也可以用Looper做参数,这样生成的Handler对应的是Looper所在的线程。可以通过Looper.myLooper()获取当前线程的Looper。调用Handler对象的sendMessage(Message)函数可以向对应线程发送消息。
例:
public class ThreadMain extends Activity {
    /** Called when the activity is first created. */
private final static int MSG_BEGIN=1;
private final static int MSG_END=2;
    private ProgressDialog dialog=null;
    private Button bt=null;
    private Handler handle=new Handler(){
    public void handleMessage(Message msg){
       switch(msg.what){
       case MSG_BEGIN:
        showProgress();
        break;
       
       case MSG_END:
        dismissProgress();
        alertDialog();
        break;
       }
       super.handleMessage(msg);
    }
    };
   
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        bt=(Button)findViewById(R.id.bt);
        bt.setOnClickListener(new OnClickListener(){
    @Override
    public void onClick(View v) {
     // TODO Auto-generated method stub
     doThread();
    }});
    }
   
    private void doThread(){
    new Thread(){
       public void run(){
        try{
         sendMsg(MSG_BEGIN);
         Thread.sleep(4000);
         sendMsg(MSG_END);
        }catch(Exception e){
         e.printStackTrace();
        }
       }
    }.start();
    }
   
    private void sendMsg(int m){
    Message msg=new Message();
    msg.what=m;
    handle.sendMessage(msg);
    }
   
    private void showProgress(){
    dialog=ProgressDialog.show(this, "", "正在……", false);
    }
   
    private void dismissProgress(){
    dialog.dismiss();
    }
   
    private void alertDialog(){
    new AlertDialog.Builder(ThreadMain.this).setMessage("完成操作")
    .setNegativeButton("确定", new DialogInterface.OnClickListener(){
    @Override
    public void onClick(DialogInterface dialog, int which) {
     // TODO Auto-generated method stub
    }}).show();
    }
}

3.后台异步任务:
Adnroid的AsyncTask类实现了后台任务的处理,即那些通过“ 提交信息->后台处理->返回结果”步骤完成的任务。
例:
class PageTask extends AsyncTask<String, Integer, String> {

        /* 对应类型AsyncTask< doInBackground函数参数类型 , onProgressUpdate函数参数类型 , doInBackground函数返回结果类型>
       */

        @Override
        protected String doInBackground(String... params) {
/*后台函数,可以放心的调用主线程的成员,params是不定参数 , 取决于执行时的输入,可以用params[0]、params[1]等方式引用*/
            return null;
        }
        @Override
        protected void onCancelled() {
            super.onCancelled();
        }
        @Override
        protected void onPostExecute(String result) {
            // 返回结果,result为doInBackground的返回值
            message.setText(result);
        }
        @Override
        protected void onPreExecute() {
            // 任务启动,可以在这里显示一个对话框,这里简单处理
            message.setText(R.string.task_started);
        }
        @Override
        protected void onProgressUpdate(Integer... values) {
            // 更新进度
            message.setText(values[0]);
        }
    }


启动方式:
PageTask task = new PageTask();
task.execute(str1,str2);