Android编程中,发生android.view.ViewRoot$CalledFromWrongThreadException异常的解决方案

在Android平台下,进行多线程编程时,经常需要在主线程之外的一个单独的线程中进行某些处理,然后更新用户界面显示。但是,在主线线程之外的线程中直接更新页面显示的问题是:系统会报这个异常,android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views. (只有原始创建这个视图层次(view hierachy)的线程才能修改它的视图(view)。)。

  也就是说必须在一般必须在程序的主线程(也就是ui)线程中进行更新界面显示的工作。可以采用下面的方法之一来解决:

  解决方案1:在Activity.onCreate(Bundle savedInstanceState)中创建一个Handler类的实例, 在这个Handler实例的handleMessage回调函数中调用更新界面显示的函数。例如:

  view plaincopy to clipboardprint?

  public class ExampleActivity extends Activity {

  Handler h = null;

  @Override

  public void onCreate(Bundle savedInstanceState){

  h = new Handler(){

  @Override

  public void handleMessage(Message msg){

  // call updateGui method.

  }

  };

  }

  }

  public class ExampleActivity extends Activity {

  Handler h = null;

  @Override

  public void onCreate(Bundle savedInstanceState){

  h = new Handler(){

  @Override

  public void handleMessage(Message msg){

  // call updateGui method.

  }

  };

  }

  }

  在其它的函数中,利用 send族或post族函数向这个h发送或邮寄消息即可。

  解决方案2:利用Activity.runOnUiThread(Runnable)

  把更新ui的代码创建在Runnable中,然后在需要更新ui时,把这个Runnable对象传给Activity.runOnUiThread(Runnable)。 这样Runnable对像就能在ui程序中被调用。

 

http://www.cnmsdn.com/html/201009/1284080180ID7930.html

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android开发,当从非UI线程访问UI组件时,就会抛出 `android.view.ViewRootImpl$CalledFromWrongThreadException` 异常。这是因为AndroidUI组件不是线程安全的,只能在主线程更新UI。 要解决这个问题,有以下几种方法: 1. 使用 `runOnUiThread()` 方法 `runOnUiThread()` 方法可以在主线程执行一个Runnable对象。因此,如果需要在非UI线程更新UI,可以使用 `runOnUiThread()` 方法来实现。 例如: ``` runOnUiThread(new Runnable() { public void run() { // 在这里更新UI } }); ``` 2. 使用 `Handler` 类 `Handler` 类可以用于在不同的线程之间传递消息和任务。通过在主线程创建一个 `Handler`,并将其传递给子线程,就可以在子线程更新UI。 例如: ``` Handler handler = new Handler(Looper.getMainLooper()); // 在子线程发送消息 new Thread(new Runnable() { public void run() { Message message = handler.obtainMessage(); message.what = 1; handler.sendMessage(message); } }).start(); // 在主线程处理消息 handler = new Handler(Looper.getMainLooper()) { public void handleMessage(Message msg) { // 在这里更新UI } }; ``` 3. 使用 `AsyncTask` 类 `AsyncTask` 类可以用于在后台线程执行耗时操作,并在主线程更新UI。在 `AsyncTask` 的 `doInBackground()` 方法执行耗时操作,在 `onPostExecute()` 方法更新UI。 例如: ``` private class MyTask extends AsyncTask<Void, Void, Void> { protected Void doInBackground(Void... params) { // 在这里执行耗时操作 return null; } protected void onPostExecute(Void result) { // 在这里更新UI } } // 在主线程执行异步任务 new MyTask().execute(); ``` 总之,无论使用哪种方法,都需要保证更新UI的操作在主线程执行。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值