Android UI更新
Android应用程序通常运行在一个dalvik虚拟机进程中,进程开始的时候会启动一个主线程(MainThread),主线程负责处理和UI相关的事件,如:用户的按键事件,因此主线程通常又叫UI线程。而由于Android采用UI单线程模型,所以只能在主线程中对UI元素进行操作。如果在非UI线程直接对UI进行了操作,则会报错:CalledFromWrongThreadException:only the original thread that created a view hierarchy can touch its views。
Android提供了消息循环的机制,利用这个机制可以实现线程间的通信,从而达到更新UI的操作。
为了保证UI线程能够及时响应用户的操作,应尽量避免在UI线程中进行费时操作。如果UI线程被阻塞5秒,则系统会弹出ANR对话框,等待用户进一步操作。
Message Queue是一个消息队列,用来存放通过Handler发送的消息。Android在第一启动程序时会默认会为UI thread创建一个关联的消息队列,可以通过Looper.myQueue()得到当前线程的消息队列,用来管理程序的一些上层组件,activities,broadcast receivers 等,你可以在自己的子线程中创建Handler与UI thread通讯。
Handler会向message queue通过两种方法发送消息:send或post。这两种消息都会插在message queue队尾并按先进先出执行,但通过这两种方法发送的消息执行的方式略有不同:1)通过send发送的是一个message对象, 会被handler的 handleMessage()函数处理;2)而通过post方法发送的是一个runnable对象,则会自己执行。
UI线程图形界面中的view可通过post方法向GUI线程的message queue投递一个runnable。对于除UI线程以外的其他线程,创建时缺省并没有message queue,而对于UI线程,则可以直接(比如在onCreate)创建一个handler并重载handleMessage,省去创建message queue的过程。
如果在UI线程中,把费时操作放到新开启的线程中,当操作完成时,通过Handler来统一对相关UI进行更新,这是一个可行且易于理解的处理方法。
代码结构如下:
public class RefreshUIActivity extends Activity implements OnClickListener { private static final int ACTION_BUTTON_1 = 1; private static final int ACTION_BUTTON_2 = 2; private static final int ACTION_BUTTON_3 = 3; private Button button1; private Button button2; private Button button3; private Handler mHandler = new Handler(new Callback() { @Override public boolean handleMessage(Message msg) { switch (msg.what) { case ACTION_BUTTON_1: button1.setText("Button1 " + msg.what); break; case ACTION_BUTTON_2: button2.setText("Button2 " + msg.what); break; case ACTION_BUTTON_3: button3.setText("Button3 " + msg.what); break; default: break; } return false; } }); @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); button1 = (Button) findViewById(R.id.button1); button2 = (Button) findViewById(R.id.button2); button3 = (Button) findViewById(R.id.button3); button1.setOnClickListener(this); button2.setOnClickListener(this); button3.setOnClickListener(this); } @Override public void onClick(View v) { switch (v.getId()) { case R.id.button1: handleButton1(); break; case R.id.button2: handleButton2(); break; case R.id.button3: handleButton3(); break; default: break; } } private void handleButton3() { new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(3 * 1000); } catch (InterruptedException e) { e.printStackTrace(); } mHandler.sendEmptyMessage(ACTION_BUTTON_3); } }).start(); } private void handleButton2() { new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(3 * 1000); } catch (InterruptedException e) { e.printStackTrace(); } mHandler.sendEmptyMessage(ACTION_BUTTON_2); } }).start(); } private void handleButton1() { new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(3 * 1000); } catch (InterruptedException e) { e.printStackTrace(); } mHandler.sendEmptyMessage(ACTION_BUTTON_1); } }).start(); } }
说明:上面仅仅是个人喜欢的代码结构罢了,呵呵!
多说一句:当然还有其它的机制可以完成更新UI的操作,在此不再赘述!=^_^=