Android关于线程更新UI的方法

Android关于线程更新UI的方法

在一个Android 程序开始运行的时候,会单独启动一个Process。默认的情况下,所有这个程序中的Activity或者Service(Service和 Activity只是Android提供的Components中的两种,除此之外还有Content Provider和Broadcast Receiver)都会跑在这个Process。
   一个Android 程序默认情况下也只有一个Process,但一个Process下却可以有许多个Thread。
   在这么多Thread当中,有一个Thread,我们称之为UI Thread。UI Thread在Android程序运行的时候就被创建,是一个Process当中的主线程Main Thread,主要是负责控制UI界面的显示、更新和控件交互。在Android程序创建之初,一个Process呈现的是单线程模型,所有的任务都在一个线程中运行。因此,我们认为,UI Thread所执行的每一个函数,所花费的时间都应该是越短越好。而其他比较费时的工作(访问网络,下载数据,查询数据库等),都应该交由子线程去执行,以免阻塞主线程。
     那么,UI Thread如何和其他Thread一起工作呢?常用方法是:
     诞生一个主线程的Handler物件,当做Listener去让子线程能将讯息Push到主线程的Message Quene里,以便触发主线程的handlerMessage()函数,让主线程知道子线程的状态,并在主线程更新UI。
       例如,在子线程的状态发生变化时,我们需要更新UI。如果在子线程中直接更新UI,通常会抛出下面的异常:
      11-07 13:33:04.393: ERROR/JavaBinder(1029):android.view.ViewRoot$CalledFromWrongThreadException:Only the original thread that created a view hierarchy can touch its views.
       意思是,无法在子线程中更新UI。为此,我们需要通过Handler物件,通知主线程Ui Thread来更新界面。
        如下,首先创建一个Handler,来监听Message的事件:
       private final int UPDATE_UI = 1;
       private Handler mHandler = new MainHandler();
      private class MainHandler extends Handler {
         @Override
             public void handleMessage(Message msg) {
             switch (msg.what) {
                 case UPDATE_UI: {
                Log.i("TTSDeamon", "UPDATE_UI");
                showTextView.setText(editText.getText().toString());
                ShowAnimation();
                     break;
                 }
                 default:
                     break;
             }
         }
       }
     或者
      private Handler mHandler = new Handler(){
         @Override
             public void handleMessage(Message msg) {
             switch (msg.what) {
                 case UPDATE_UI: {
                Log.i("TTSDeamon", "UPDATE_UI");
                showTextView.setText(editText.getText().toString());
                ShowAnimation();
                     break;
                 }
                 default:
                     break;
             }
          }
      }
       当子线程的状态发生变化,则在子线程中发出Message,通知更新UI。
       mHandler.sendEmptyMessageDelayed(UPDATE_UI, 0);
      在我们的程序中,很多Callback方法有时候并不是运行在主线程当中的,所以如果在Callback方法中更新UI失败,也可以采用上面的方法。
第二篇
Android的UI设计与后台线程交互

本文将讨论Android应用程序的线程模型以及如何使用线程来处理耗时较长的操作,而不是在主线程中执行,保证用户界面(UI)的流畅运行。
     在 Android的UI设计方面我们讲过“Android UI”设计官方教程。本文将讨论Android应用程序的线程模型以及如何使用线程来处理耗时较长的操作,而不是在主线程中执行,保证用户界面(UI)的流畅运行。本文还将阐述一些用户界面(UI)中与线程交互的API。
     UI用户界面线程
当应用程序启动时,系统会为应用程序创建一个主线程(main)或者叫UI线程,它负责分发事件到不同的组件,包括绘画事件。完成你的应用程序与Android UI组件交互。
例如,当您触摸屏幕上的一个按钮时,UI线程会把触摸事件分发到组件上,更改状态并加入事件队列,UI线程会分发请求和通知到各个组件,完成相应的动作。
单线程模型的性能是非常差的,除非你的应用程序相当的简单,特别是当所有的操作都在主线程中执行,比如访问网络或数据库之类的耗时操作将会导致用户界面锁定,所有的事件将不能分发,应用程序就像死了一样,更严重的是当超过5秒时,系统就会弹出“应用程序无响应”的对话框。
      如果你想看看什么效果,可以写一个简单的应用程序,在一个Button的OnClickListener中写上Thread.sleep(2000),运行程序你就会看到在应用程序回到正常状态前按钮会保持按下状态2秒,当这种情况发生时,您就会感觉到应用程序反映相当的慢。
总之,我们需要保证主线程(UI线程)不被锁住,如果有耗时的操作,我们需要把它放到一个单独的后台线程中执行。
下面是一个点击按钮后下载一个图片,同时显示到界面的ImageView上的例子:
public void onClick(View v) {  
  new Thread(new Runnable() {  
    public void run() {  
      Bitmap b = loadImageFromNetwork();  
      mImageView.setImageBitmap(b);  
    }  
  }).start();  
}  
起初,上面的代码似乎是一个很好的解决方案,因为它不会锁住用户界面线程。然面不幸的是,它违反了用户界面单线程模型:Android的用户界面工具包不是线程安全的,只能在UI线程中操作它,在上面的代码中,你在一个工作线程中调用mImageView.setImageBitmap(b)时,将会发生意想不到的错误,这种错误是非常难跟踪和调试的。
Android提供了几种方法来从其他线程访问UI线程。您可能已经熟悉他们了,下面是一个较全面的列表:
Activity.runOnUiThread(Runnable)  
View.post(Runnable)  
View.postDelayed(Runnable, long)  
Handler  
您可以使用这些类和方法中的任何一种纠正前面的代码示例:
public void onClick(View v) {  
  new Thread(new Runnable() {  
    public void run() {  
      final Bitmap b = loadImageFromNetwork();  
      mImageView.post(new Runnable() {  
        public void run() {  
          mImageView.setImageBitmap(b);  
        }  
      });  
    }  
  }).start();  

不幸的是,这些类和方法也往往使你的代码更复杂,更难以阅读。更糟糕的是,它需要频繁执行复杂的操作界面更新。
为了解决这个问题,1.5和更高版本的Android平台提供了一个实用类称为AsyncTask,简化了长时间运行的任务,需要与用户界面的交互。
类似AsyncTask的一个类UserTask也可用于Android 1.0和1.1版本,它提供了完全相同的API,所有您需要做的是把它的源代码复制到你的应用程序中。
AsyncTask的目标是要为你的线程提供管理服务,我们前面的例子可以很容易的用AsyncTask来改写:
public void onClick(View v) {  
  new DownloadImageTask().execute("http://www.ideasandroid.com/image.png");  
}  
private class DownloadImageTask extends AsyncTask<String, Void,Bitmap> {  
     protected Bitmap doInBackground(String... urls) {  
         return loadImageFromNetwork(urls[0]);  
     }  
     protected void onPostExecute(Bitmap result) {  
         mImageView.setImageBitmap(result);  
     }  
 } 
正如你所看到的,我们必须通过继承AsyncTask类来使用它,非常重要的一点是:AsyncTask必须在UI线程中实例化它,并且只能执行一次。
以下是AsyncTask的简要使用方法:
◆您可以指定三个参数类型,泛型参数,进度值(执行过程中返回的值)和最终值(执行完返回的值)。
◆该方法doInBackground()自动执行工作线程(后台线程)
◆onPreExecute(),onPostExecute()和onProgressUpdate()都是在UI线程调用
◆由doInBackground返回的值()发送到onPostExecute()
◆您可以在执行doInBackground()时调用publishProgress()然后在UI组程中执行onProgressUpdate()。
◆您可以从任何线程随时取消任务
不管你是否使用AsyncTask,时刻牢记单一线程模型的两条规则:
1、不要锁住用户界面。
2、确保只在UI线程中访问Android用户界面工具包中的组件。
AsyncTask只是可以让你更容易地做这些事情。

Android关于线程更新UI的方法

时间: 2024-12-25 19:18:42

Android关于线程更新UI的方法的相关文章

Android子线程更新UI主线程方法之Handler

背景: 我们开发应用程序的时候,处于线程安全的原因子线程通常是不能直接更新主线程(UI线程)中的UI元素的,那么在Android开发中有几种方法解决这个问题,其中方法之一就是利用Handler处理的. 下面说下有关Handler相关的知识. 多线程一些基础知识回顾:在介绍Handler类相关知识之前,我们先看看在Java中是如何创建多线程的方法有两种:通过继承Thread类,重写Run方法来实现通过继承接口Runnable实现多线程 具体两者的区别与实现,看看这篇文章中的介绍:http://de

android中子线程更新UI的方式浅析

一.为何写作此文 ??你是不是经常看到很多书籍中说:不能在子线程中操作ui,不然会报错.你是不是也遇到了如下的疑惑(见下面的代码): @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); tv = (TextView) findViewById(R.id.tv); Threa

android四种更新UI的方法

笔记: // 使用handler.post(Runnable)更新UI public void updateUI_Fun1() { new Thread() { public void run() { Handler handler = new Handler(); handler.post(new Runnable() { @Override public void run() { textView.setText("this is update content"); } }); }

Android 子线程 更新 UI 界面 总结

package com.jrhcode.morethreadtest; import java.util.concurrent.Executors; import android.os.AsyncTask; import android.os.Bundle; import android.os.Handler; import android.app.Activity; import android.view.Menu; import android.widget.TextView; import

Android 子线程更新UI 异常

众所周知,Android是不可以在子线程中直接更新UI的,需要借助Handler或者View.post(Runnable runnable)或者runOnUIThread(Runnable runnable)将更新的代码切入到主线程中去实现UI更新. 我们来试一下,在Activity的 onCreate中直接在新线程中去更新一个TextView的文本,结果发现,WHAT?竟然没有抛出异常,更新也成功了? 这是因为在onCreate中尚未完成View的绘制,此时TextView中有个变量mLayo

子线程更新UI界面的2种方法

一.一般我们都会在子线程完成一些耗时的操作. 1.Android中消息机制: 2.知识点: Message:消息,其中包含了消息ID,消息处理对象以及处理的数据等,由MessageQueue统一列队,终由Handler处理.                      Handler:处理者,负责Message的发送及处理.使用Handler时,需要实现handleMessage(Message msg)方法来对特定的Message进行处理,例如更新UI等.                    

Qt 子线程更新Ui

最近做练习,写一个Qt版的飞机大战,需要用子线程更新UI,发现Qt子线程不能更新Ui,否则程序会崩溃.在网上百度了下,说是需要在子线程自定义信号,然后在线程回调的run()函数里发射信号,主线程连接信号和槽,然后在槽函数里面更新UI.虽然最后发现这个办法对我写飞机大战没有啥帮助,但是感觉这个办法还是需要总结下来的.(最后我是用的定时器更新的UI界面) 废话不多说,下面是子线程更新UI的方法: 第一步:写一个线程类,继承自QThread 第二步:自定义信号 class CMyThread : pu

android子线程中更新UI的方法

在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法: 参考:Android子线程 方法一:用Handler 1.主线程中定义Handler: Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) { super.handleMessage(msg); switch (msg.what) { case 0: //

Android在非UI线程中更新UI的方法

1.使用Thread+Handler实现非UI线程更新UI界面 private MyHandler mHandler = new MyHandler(); ...... mHandler.sendResult(MyHandler.UPDATE_VIEW, null); ...... private class MyHandler extends Handler{ private static final int UPDATE_VIEW = 0; @Override public void han