当前位置:   article > 正文

Android中使用Handler机制更新UI的三种解决方案_handler更新ui偶尔失败

handler更新ui偶尔失败

使用Handler其实不得不与Android中的线程或者说Java中的多线程扯上关系。本篇文章只会用到最基本的线程使用,不会涉及太难的,关于Android的线程我们以后再讨论。在Android中每新建一个Activity,该Activity(理解为界面)就是一个线程,是一个主线程,也称之为UI线程。主线程可以更新界面元素,不会出现问题。每当新建一个线程new Thread,该线程就是一个子线程,Android规定子线程是不能直接更新UI界面的,否则会出问题。

下面将会使用Android提供的Handler机制在子线程中更新UI.分别使用3种方式,3种方式相互联系又有区别。

(1)Handler.post()方式

该方法使用Handler中的post()方法更新UI,这里会涉及Thread,Runnable两个线程的概念,先暂且忽略吧。该方法比较简单,容易理解。先贴上代码:

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public class MainActivity extends Activity {
 
     private TextView text;
     private Handler handler = new Handler();
 
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.activity_main);
         text = (TextView) findViewById(R.id.id_text);
 
         /**
          * new Thread()在该线程中实现你具体的业务逻辑,比如网络请求,耗时操作等等;
          * new Thread()是一个子线程,是非UI线程,如果在该线程中需要更新界面,则需要使用Handler;
          *
          */
         new Thread() {
             @Override
             public void run() { //在run()方法实现业务逻辑;
                 //...
 
                 //更新UI操作;
                 handler.post( new Runnable() {
                     @Override
                     public void run() {
                         text.setText(使用Handler更新了界面);
                     }
                 });
 
             }
         }.start();
     }
}

注意看其中的注释。开发者的对线程的业务逻辑操作写在Thread.run()方法中,更新UI的操作写到Runnable.run()方法中。实现效果如下:

 

\

 

(2)Handler.post(),内部类实现Runnable接口方式

该方法与方法1非常像,只是新建了一个内部类,并且实现了Runnable接口,然后在post()方法中不用new一个匿名内部类了。相对来说逻辑上更加清楚。思路同方法1.贴上代码如下:

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class MainActivity extends Activity {
 
     private TextView text;
     private Handler handler = new Handler();
 
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.activity_main);
         text = (TextView) findViewById(R.id.id_text);
         final MyRunnable myRunnable = new MyRunnable(); //定义MyRunnable的对象;
 
         new Thread() {
 
             @Override
             public void run() {
                 handler.post(myRunnable); //调用Handler.post方法;
             }
         }.start();
     }
 
     class MyRunnable implements Runnable { //内部类实现Runnable接口;
 
         @Override
         public void run() { //还是在Runnable重写的run()方法中更新界面;
             text.setText(使用Handler更新了界面);
         }
     }
}

注意提醒一点:Thread必须调用start()方法,否则线程不会被执行。实现效果如下:

 

\

 

(3)sendMessage(),handleMessage()方式

在Handler中有两个非常重要的方法,sendMessage()和handleMessage()方法,sendMessage()方法用于在线程中向Handler发送一个消息,handleMessage()用于捕获该消息,并且更新UI.代码如下:

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public class MainActivity extends Activity {
 
     private TextView text;
     private Handler handler = new Handler() {
 
         @Override
         public void handleMessage(Message msg) {
             switch (msg.what) {
                 case 1 :
                     text.setText(使用Handler更新了界面);
                     break ;
             }
         }
     };
 
     @Override
     protected void onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.activity_main);
         text = (TextView) findViewById(R.id.id_text);
 
         new Thread() {
 
             @Override
             public void run() {
                 //...你的业务逻辑;
                 Message message = new Message(); //发送一个消息,该消息用于在handleMessage中区分是谁发过来的消息;
                 message.what = 1 ;
                 handler.sendMessage(message);
             }
         }.start();
     }
}

实现效果同上述两个方法:

 

\

 

至此,已经基本实现了在子线程中使用Handler更新了UI界面。能满足最基本的开发需要。


handler.post()和handler.sendMessage()有什么区别?

我们都知道Handler中的post方法,并且也是经常使用它

handler.post(new Runnable(){
@Override
public void run() {
//do something

}});
用它可以更新一个组件的内容,我们也知道Hanlder中也有一个handler.sendMessage(Message msg)方法,这两个方法有什么区别呢?先看一下Message类中定义一个私有的变量:Runnable callback;
再来看一下handler.post(Runnable callback)方法的源码:
public final boolean post(Runnable r) {
return sendMessageDelayed(getPostMessage(r), 0);
}

再看一下sendMessageDelayed的源码:
public final boolean sendMessageDelayed(Message msg, long delayMillis)
{
if (delayMillis < 0) {
delayMillis = 0;
}
return sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis);
}

这里面有个关键就是方法getPostMessage(r)这个方法,他将Runnable转成一个Message,他内部到底干了什么呢?看一下他的源码:
private final Message getPostMessage(Runnable r) {
Message m = Message.obtain();
m.callback = r;
return m;
}

这里面就是将Runnable转化成一个Message,其他看他的代码很简单,就是先获取一个空消息Message.obtain(),然后将Message中的callback的值设置成Runnable,这时候就了解到了Message中的callback的作用了!
同时也了解一下View.post(Runnable r)方法的作用:看一下实例代码:
final Button btn = (Button)findViewById(R.id.btn);
btn.post(new Runnable(){
@Override
public void run() {
btn.setText("不是好人");
}
}); 
}

上面的代码就是更新btn中的内容,同样下面的代码也可以达到这种效果:
Handler handler = new Handler();
final Button btn = (Button)findViewById(R.id.btn);
handler.post(new Runnable(){
@Override
public void run() {
btn.setText("不是好人");
}
}); 
}

不同是这个是用handler.post方法,一个是用View.post方法,现在来看一下View.post方法的源代码:
public boolean post(Runnable action) {
Handler handler;
AttachInfo attachInfo = mAttachInfo;
if (attachInfo != null) {
handler = attachInfo.mHandler;
} else {
// Assume that post will succeed later
ViewRootImpl.getRunQueue().post(action);
return true;
}
return handler.post(action);
}

方法中主要的功能代码就是attachInfo.mHandler,获取当前线程的hanlder,和我们在一个线程中定义一个Handler的效果是一样的
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/465217?site
推荐阅读
相关标签
  

闽ICP备14008679号