使用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的效果是一样的