通过使用Handler在其它线程里实现UI的更新
Handler主要用来更新界面,我们知道只有主线程才有权限去更新界面。耗时操作也不应该放在主线程,不然就会出现著名的ANR错误.还有网络请求也不能放在主线程执行。
Handler有许多的方法:
- handlerMessage(Message msg):用于处理消息的方法
- hasMessage(int what):检查消息队列是否包含what属性为指定值的消息.
- hanMessage(int what,Object object):检查消息队列中是否包含what属性为定值且object属性为指定对象的消息
- sendEmptyMessage(int what):发送空消息
- sendEmptyMessageDelayed(int what, long delayMillis):指定多少毫秒之后发送空消息
- sendMessage(Messasge msg):立即发送消息.
- sendMessageDelayed(Message msg, long delayMisllis) 指定多少毫秒之后发送消息
闲话不多说,要学会还是得撸代码。
public class MainActivity extends Activity {
private TextView textView;
private Button btn;
private Handler handler = new Handler(){
public void handleMessage(android.os.Message msg) {
if(msg.what == 0x123){
textView.setText("Data changed");
}
};
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
textView = (TextView) this.findViewById(R.id.textView);
btn = (Button) this.findViewById(R.id.button1);
btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
new Thread(){
@Override
public void run() {
super.run();
handler.sendEmptyMessage(0x123);
}
}.start();
}
});
}
}
可以看到我们点击了Button之后,TextView的文本内容变成了”Data changed”,如果在new Thread(){}里面直接对TextView进行操作则会出现以下错误。
在其他线程通过Handler对UI进行操作的第2种方式:
我们使用Handler.post(Runnable)或者Handler.post(Runnable, delaymillis)来控制UI。
public class MainActivity extends Activity {
private TextView textView;
private Button btn;
private Handler handler = new Handler(){
public void handleMessage(android.os.Message msg) {
String info = (String) msg.obj;
textView.setText(info);
};
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
textView = (TextView) this.findViewById(R.id.textView);
btn = (Button) this.findViewById(R.id.button1);
btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
new Thread(){
@Override
public void run() {
super.run();
handler.postDelayed(new Runnable() {
@Override
public void run() {
//获取Handler自带的Message对象,Message的成员变量有整形的 arg1 arg2 还有Object。 Message.sendToTaget()方法将数据发送给Handler
Message msg = handler.obtainMessage();
msg.obj = "Data changed";
msg.sendToTarget();
}
}, 5000);
}
}.start();
}
});
}
}
第三种方式:`
public class MainActivity extends Activity {
private TextView textView;
private Button btn;
private final Handler myHandler = new Handler(this.getMainLooper()){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
Bundle bundle = msg.getData();
int age = bundle.getInt("age");
String name = bundle.getString("name");
System.out.println("....>>>>>>>"+name+age);
textView.setText(name+age);
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
textView = (TextView) this.findViewById(R.id.textView);
btn = (Button) this.findViewById(R.id.button1);
btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
new Thread(){
@Override
public void run() {
super.run();
Message msg = myHandler.obtainMessage();
//往包裹装东西,Handler就相当与一个邮递员.Message就相当于邮包。
Bundle bundle = new Bundle();
bundle.putInt("age",20);
bundle.putString("name","Huber");
msg.setData(bundle);
msg.sendToTarget();
}
}.start();
}
});
}
}
这种方式其实也是往主线程的消息队列发送消息.this.getMainLooper()就是获得主线程的消息队列.而myHandler正是运行在主线程中的。所以有更新UI界面的权限。sendToTarget()方法,看源码可以知道,其实最终还是调用Handler的sendMessage()的方法。
附上XML的代码:
`<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world" />
<Button
android:id="@+id/button1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@+id/textView"
android:layout_marginLeft="51dp"
android:layout_marginTop="125dp"
android:layout_toRightOf="@+id/textView"
android:text="Button" />
</RelativeLayout>
`
效果图就不发了,你们可以自己去验证.毕竟代码量不多。
至于Handler,MessageQueen,Looper的关系,我会在下一个博客对它们进行学习,以及表达我对他们的理解。
`