❶ android子线程更新UI的方法
在进行Android开发时,需确保UI操作仅在主线程中执行,以避免应用性能问题。若需在子线程中更新UI,可采取以下策略:
一种常用方法是使用Handler。在主线程定义Handler实例,利用其发送消息与回调机制,在子线程执行完毕后更新UI。
另一种选择是AsyncTask,一个轻量级的异步任务类。它在子线程中执行后台任务,通过回调方法如doInBackground、onProgressUpdate和onPostExecute传递结果至主线程,进而更新UI。
还有一种方法是Activity的runOnUiThread方法。通过Activity对象调用此方法,可将更新UI的代码放入回调中执行。
最后,可利用View.post(Runnable r)方法,将更新UI的Runnable对象传递给视图,由视图在合适时机执行该操作。
尽管以上方法允许子线程更新UI,频繁操作仍可能对应用性能造成负面影响。因此,应谨慎使用。
❷ 如何在子线程中更新ui
在子线程中更新UI:由于安卓更新UI元素必须在主线程中,否则就会出现异常。有时候我们必须在子线程里去执行一些耗时任务,然后根据任务的执行结果来更新相应的UI控件,比如:执行网络请求,需要将传回的数据更新到UI上。
操作方法:
一、在主线程中new出Handler的实例handler,在Handler的handleMessage()方法中传回的Message对象属性进行判断,当满足要求时,执行UI更新的操作。
在子线程的run()方法内,1、实例化Message对象message;2、对象调用Message的what变量,设置message的属性;3、对象调用sendMessage()方法,将Message对象发送出去。
private Handler handler = new Handler() {
public void handleMessage(Message msg) {
switch (msg.what) {
case UPDATE_TEXT:
// 在这里可以进行UI操作
text.setText("Nice to meet you");
break;
default:
break;
}
}
new Thread(new Runnable() {
@Override
public void run() {
Message message = new Message();
message.what = UPDATE_TEXT;
handler.sendMessage(message); // 将Message对象发送出去
}
}).start();
❸ android通过Handler使子线程更新UI
在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法。
一. 引言
首先来看一下android中消息机制:
专业术语:
Message:消息,其中包含了消息ID,消息处理对象以及处理的数据等,由MessageQueue统一列队,终由Handler处理。
Handler:处理者,负责Message的发送及处理。使用Handler时,需要实现handleMessage(Message msg)方法来对特定的Message进行处理,例如更新UI等。
MessageQueue:消息队列,用来存放Handler发送过来的消息,并按照FIFO规则执行。当然,存放Message并非实际意义的保存,而是将Message以链表的方式串联起来的,等待Looper的抽取。
Looper:消息泵,不断地从MessageQueue中抽取Message执行。因此,一个MessageQueue需要一个Looper。
Thread:线程,负责调度整个消息循环,即消息循环的执行场所。
二. 方法
1. 用Handler
(1)主线程中定义Handler:
java代码:
[java]view plain
HandlermHandler=newHandler(){
@Override
publicvoidhandleMessage(Messagemsg){
super.handleMessage(msg);
switch(msg.what){
case0:
<spanstyle="color:#009900;">//完成主界面更新,拿到数据</span>
Stringdata=(String)msg.obj;
updateWeather();
textView.setText(data);
break;
default:
break;
}
}
};
(2)子线程发消息,通知Handler完成UI更新:
java代码:
privatevoipdateWeather(){
newThread(newRunnable(){
@Override
publicvoidrun(){
<spanstyle="color:#009900;">//耗时操作,完成之后发送消息给Handler,完成UI更新;</span>
mHandler.sendEmptyMessage(0);
<spanstyle="color:#33cc00;">//需要数据传递,用下面方法;</span>
Messagemsg=newMessage();
msg.obj="数据";<spanstyle="color:#33cc00;">//可以是基本类型,可以是对象,可以是List、map等;</span>
mHandler.sendMessage(msg);
}
}).start();
}
注意:Handler对象必须定义在主线程中,如果是多个类直接互相调用,就不是很方便,需要传递content对象或通过接口调用。
2.用Activity对象的runOnUiThread方法更新
在子线程中通过runOnUiThread()方法更新UI:
java代码:
newThread(){
publicvoidrun(){
<spanstyle="color:#009900;">//这儿是耗时操作,完成之后更新UI;</span>
runOnUiThread(newRunnable(){
@Override
publicvoidrun(){
<spanstyle="color:#009900;">//更新UI</span>
imageView.setImageBitmap(bitmap);
}
});
}
}.start();
如果在非上下文类中,可以通过传递上下文实现调用:
java代码:
Activityactivity=(Activity)imageView.getContext();
activity.runOnUiThread(newRunnable(){
@Override
publicvoidrun(){
imageView.setImageBitmap(bitmap);
}
});
注意:这种方法使用比较灵活,但如果Thread定义在其他地方,需要传递Activity对象。
3.
View.post(Runnable r)
java代码:
imageView.post(newRunnable(){
@Override
publicvoidrun(){
imageView.setImageBitmap(bitmap);
}
});
这种方法更简单,但需要传递要更新的View过去。
总结:UI的更新必须在主线程中完成,所以不管上述那种方法,都是将更新UI的消息发送到了主线程的消息对象,让主线程做处理。
❹ 在多线程中,子线程更新主线程ui有哪些方法及注意点
Android
UI多线程Androidthread工作
在一个Android 程序开始运行的时候,会单独启动一个Process。默认的情况下,所有这个程序中的Activity或者Service(Service和 Activity只是Android提供的Components中的两种,除此之外还有Content Provider和Broadcast Receiver)都会跑在这个Process。
一个Android 程序默认情况下也只有一个Process,但一个Process下却可以有许多个Thread。在这么多Thread当中,有一个Thread,我们称之为UI Thread。UI Thread在Android程序运行的时候就被创建,是一个Process当中的主线程Main Thread,主要是负责控制UI界面的显示、更新和控件交互。在Android程序创建之初,一个Process呈现的是单线程模型,所有的任务都在一个线程中运行。因此,我们认为,UI Thread所执行的每一个函数,所花费的时间都应该是越短越好。而其他比较费时的工作(访问网络,下载数据,查询数据库等),都应该交由子线程去执行,以免阻塞主线程。
那么,UI Thread如何和其他Thread一起工作呢?常用方法是:
诞生一个主线程的Handler物件,当做Listener去让子线程能将讯息Push到主线程的Message Quene里,以便触发主线程的handlerMessage()函数,让主线程知道子线程的状态,并在主线程更新UI。
例如,在子线程的状态发生变化时,我们需要更新UI。如果在子线程中直接更新UI,通常会抛出下面的异常:11-07 13:33:04.393: ERROR/JavaBinder(1029):android.view.ViewRoot$:Only the original thread that created a view hierarchy can touch its views.
意思是,无法在子线程中更新UI。为此,我们需要通过Handler物件,通知主线程Ui Thread来更新界面。
如下,首先创建一个Handler,来监听Message的事件:
private final int UPDATE_UI = 1;private Handler mHandler = new MainHandler();private class MainHandler extends Handler {@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case UPDATE_UI: {Log.i("TTSDeamon", "UPDATE_UI");showTextView.setText(editText.getText().toString());ShowAnimation();break;}default:break;}}}
或者
private Handler mHandler = new Handler(){@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case UPDATE_UI: {Log.i("TTSDeamon", "UPDATE_UI");showTextView.setText(editText.getText().toString());ShowAnimation();break;}default:break;}}}
当子线程的状态发生变化,则在子线程中发出Message,通知更新UI。
mHandler.sendEmptyMessageDelayed(UPDATE_UI, 0);
在我们的程序中,很多Callback方法有时候并不是运行在主线程当中的,所以如果在Callback方法中更新UI失败,也可以采用上面的方法。