Android 在子线程中如何修改ui界面

转自:http://blog.csdn.net/u013174702/article/details/47775343

1. Android进程

一个应用程序被启动时,系统默认创建执行一个叫做"main"的线程。这个线程也是你的应用与界面工具包(android.widget和android.view包中的组件)交互的地方。于是main线程也被称为界面线程。

这种单线程的模式会带来低性能,除非你能正确的优化你的程序。

打 个比方:用户触摸屏幕上的一个按钮时的点击事件即向线程中派发事件,比如每一个事件为一辆车。而每一条线程都好像是一条单行线的单车道。这条车道上的车量 都排成队行驶至收费口进行事件处理。当处理的事件繁琐,比如在响应用户交互时需执行大量运算,或者像是执行网络连接、数据库请求这样耗时的操作。就会造成 拥堵,将会阻止整个界面的响应。当线程被阻塞时,就不能派发事件了。从用户的角度看,程序反应太慢了。甚至更糟的是,如果界面线程被阻塞几秒钟(大5秒钟 吧),用户就户抱怨说程序没反应了,用户可能因而退出并删掉你的应用。 此外,Andoid界面不是线程安全的。所以你绝不能在一个工作线程中操作你的界面—你只能在界面线程中管理的你的界面。所以,对于单线程模式有两个简单 的规则

1不要阻塞界面线程

2不要在界面线程之外操作界面。

 

2. 工作线程

由于上述的单线程模式,不要阻塞你的界面线程以使你的应用的界面保持响应是非常重要的,那么如果你有不能很快完成的任务,你应把它们放在另一个线程中执行(后台线程或工作线程)。

例如,下面是的代码是响应click事件,在另外一个线程中从网络获取资源文字并以TextView来显示。

[java]  view plain  copy
  1. mHandle.setOnClickListener(new OnClickListener() {  
  2.               
  3.             @Override  
  4.             public void onClick(View v) {  
  5.                 new Thread(new Runnable() {  
  6.                       
  7.                     @Override  
  8.                     public void run() {                          
  9.                                               // 耗时操作             
  10.                                                      loadNetWork();    
  11.                     mTextView.setText(来自网络的文字);                          
  12.                     }  
  13.                 });  
  14.                   
  15.             }  
  16.         });  

第一眼,这看起来能很好的工作,因为它创建了一个新线程来进行网络操作。然而它违反了第二条规则:不要在界面线程之外操作界面—它简单的在工作线程中修改了mTextView。这会导至未定义的异常出现,并且难以调试追踪。


为了能改正这个问题,Android提供了很多从其它线程来操作界面的方法。下面是可用的方法们:

1 Activity.runOnUiThread(Runnable)

2 View.post(Runnable)

3 View.postDelayed(Runnable,long)

4 Handler

5 AsyncTask

现在我们就依次使用这几个方法:

1、Activity.runOnUiThread(Runnable)

[java]  view plain  copy
  1. mHandle.setOnClickListener(new OnClickListener() {   
  2.                
  3.             @Override   
  4.             public void onClick(View v) {   
  5.                 new Thread(new Runnable() {   
  6.                        
  7.                     @Override   
  8.                     public void run() {   
  9.                         MainActivity.this.runOnUiThread(new Runnable() {   
  10.                                                    // 耗时操作              
  11.                                                           loadNetWork();     
  12.                         @Override   
  13.                             public void run() {   
  14.                                 mTextView.setText(来自网络的文字);               
  15.                             }   
  16.                         });   
  17.                            
  18.                     }   
  19.                 });   
  20.                    
  21.             }   
  22.         });   

2、  View.post(Runnable)

[java]  view plain  copy
  1. mHandle.setOnClickListener(new OnClickListener() {  
  2.               
  3.             @Override  
  4.             public void onClick(View v) {  
  5.                 new Thread(new Runnable() {  
  6.                       
  7.                     @Override  
  8.                     public void run() {  
  9.                                                      // 耗时操作             
  10.                                                      loadNetWork();        
  11.                         mTextView.post(new Runnable() {  
  12.                               
  13.                             @Override  
  14.                             public void run() {  
  15.                                 mTextView.setText(来自网络的文字);      
  16.                             }  
  17.                         });  
  18.                           
  19.                     }  
  20.                 });                                            
  21.             }  
  22.         });  

3、 View.postDelayed(Runnable,long)

[java]  view plain  copy
  1. mHandle.setOnClickListener(new OnClickListener() {  
  2.               
  3.             @Override  
  4.             public void onClick(View v) {  
  5.                 new Thread(new Runnable() {  
  6.                       
  7.                     @Override  
  8.                     public void run() {  
  9.                                                      // 耗时操作             
  10.                                                      loadNetWork();    
  11.                         mTextView.postDelayed(new Runnable() {  
  12.                               
  13.                             @Override  
  14.                             public void run() {  
  15.                                 mTextView.setText(来自网络的文字);                                  
  16.                             }  
  17.                         }, 10);  
  18.                           
  19.                     }  
  20.                 });  
  21.                   
  22.                   
  23.             }  
  24.         });  

4、 Handler(子线程调用Handler的handle.sendMessage(msg);

[java]  view plain  copy
  1. Handler handle = new Handler() {  
  2.   
  3.         @Override  
  4.         public void handleMessage(Message msg) {  
  5.             super.handleMessage(msg);  
  6.             mTextView.setText(来自网络的文字);  
  7.         }  
  8.           
  9.     };  
  10.   
  11. class MyThread extends Thread {  
  12.   
  13.         @Override  
  14.         public void run() {  
  15.              // 耗时操作             
  16.             loadNetWork();    
  17.   
  18.             Message msg = new Message();  
  19.             handle.sendMessage(msg);  
  20.             super.run();  
  21.         }  
  22.           
  23.           
  24.     }  

5、AsyncTask

主线程中:aTask ak = new aTask();
ak.execute();

然后:

[java]  view plain  copy
  1. private class aTask extends AsyncTask {   
  2.   
  3.     //后台线程执行时   
  4.     @Override   
  5.     protected Object doInBackground(Object... params) {   
  6.         // 耗时操作               
  7.         return loadNetWork();   
  8.     }   
  9.     //后台线程执行结束后的操作,其中参数result为doInBackground返回的结果   
  10.     @Override   
  11.     protected void onPostExecute(Object result) {   
  12.         super.onPostExecute(result);   
  13.         mTextView.setText(result);   
  14.     }   
  15.     }   
  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
AndroidUI线程(也称为主线程)是用于处理界面事件和更新界面的线程。在线程修改UI可能会导致ANR(应用程序无响应)错误,因为UI线程可能会被阻塞,导致应用程序无法响应用户的操作。因此,需要使用一些技术来在后台线程更新UI,以避免阻塞UI线程。 以下是一些在线程修改UI的方法: 1.使用Handler:Handler是Android的一个机制,可以用于在后台线程更新UI。可以在UI线程创建一个Handler对象,然后使用post()方法将一个Runnable对象发送到Handler对象,以便在UI线程执行该Runnable对象。示例代码如下: ```java // 创建一个Handler对象 Handler handler = new Handler(Looper.getMainLooper()); // 在后台线程发送一个Runnable对象到Handler对象 new Thread(new Runnable() { @Override public void run() { // 在后台线程更新UI handler.post(new Runnable() { @Override public void run() { // 在UI线程更新UI textView.setText("Hello World"); } }); } }).start(); ``` 2.使用runOnUiThread()方法:Activity类提供了一个runOnUiThread()方法,可以用于在后台线程更新UI。可以将一个Runnable对象传递给该方法,该方法将在UI线程执行该Runnable对象。示例代码如下: ```java new Thread(new Runnable() { @Override public void run() { // 在后台线程更新UI runOnUiThread(new Runnable() { @Override public void run() { // 在UI线程更新UI textView.setText("Hello World"); } }); } }).start(); ``` 3.使用AsyncTask:AsyncTask是Android的一个异步任务类,可以用于在后台线程执行耗时的操作,并在UI线程更新UI。可以在AsyncTask类实现doInBackground()方法来执行后台操作,在onPostExecute()方法更新UI。示例代码如下: ```java public class MyTask extends AsyncTask<Void, Void, String> { @Override protected String doInBackground(Void... voids) { // 在后台线程执行耗时操作 return "Hello World"; } @Override protected void onPostExecute(String s) { // 在UI线程更新UI textView.setText(s); } } // 在主线程执行异步任务 new MyTask().execute(); ``` 无论使用哪种方法,都需要注意在后台线程更新UI时,需要确保更新UI的操作不会阻塞UI线程,以避免ANR错误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值