android.view.ViewRoot$CalledFromWrongThreadException 异常的解决方案Activity类runOnUiThread方法

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

一般的做法是用handler机制,在子线程发一个Message个ui线程,然后线程获取message的数据更新ui控件数据。

这里着重要介绍的是第二个方案。就是activity的runOnUiThread(Runnable runnable)方法;要使用这个方法必须往子线程中传递activity引用context,和要更新的UI控件引用。
然后在子线程中实现更新逻辑,然后这个更新代码被加入到ui线程队列等待UI线程执行。

activity.runOnUiThread(new Runnable() {

    @Override
    public void run() {
                            // TODO Auto-generated method stub
                            imageView.setImageBitmap(bitmap);
                        }
                    });

//根据下载url,下载图片更新imageview控件的工具类。
//传进去的参数有下载路径url,activity引用,imageview控件引用
    public static void setPicBitmap(final String url, final Activity activity,
            final ImageView imageView) {

        new Thread() {

            @Override
            public void run() {

                try {

                    HttpURLConnection connection = (HttpURLConnection) new URL(
                            url).openConnection();
                    connection.connect();
                    InputStream iStream = connection.getInputStream();

                    final Bitmap bitmap = BitmapFactory.decodeStream(iStream);
                    // pic.setImageBitmap(bitmap);
                    activity.runOnUiThread(new Runnable() {

                        @Override
                        public void run() {
                            // TODO Auto-generated method stub
                            imageView.setImageBitmap(bitmap);
                        }
                    });
                    iStream.close();

                } catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }

            }
        }.start();

    }
Android开发中,当从非UI线程中访问UI组件时,就会抛出 `android.view.ViewRootImpl$CalledFromWrongThreadException` 异常。这是因为Android中的UI组件不是线程安全的,只能在主线程中更新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、付费专栏及课程。

余额充值