为了线程安全,android在子线程中是不允许对界面元素进行刷新的,比如
final ImageButton button = (ImageButton)this.findViewById(R.id.button1); button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { new Thread(new Runnable() { @Override public void run() { textView.setText("更改文本内容"); } }).start(); } });
是会报错:Only the original thread that created a view hierarchy can touch its views.
如果要在子进程刷新界面的话,可以使用android的handler机制,正确的使用方法如下:
final ImageButton button = (ImageButton)this.findViewById(R.id.button1); button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { new Thread(new Runnable() { @Override public void run() { button.post(new Runnable() { @Override public void run() { tv.setText("click"); } }); } }).start(); } });
此列中button.post只所以不会报错是因为他的父类View中有一个handler的引用,相当于是handler.post,
android中有一个保护机制,就是当界面在刷新时超过一段时间无响应时(20秒左右)程序就会退出,所以一般进行比较耗时的操作时必须另起一个线程,比如下载资源时,在整个过程中,我们一般同时还需要刷新下载的进度,如果是直接在子线程中刷新界面的话就会出现上面类似的错误。这个时候我们就需要使用到handler。具体列在如下:
handler = new Handler(){ public void handleMessage(Message msg) { switch (msg.what) { case 1: tv.setText("剩余下载时间:" + msg.arg1); break; case 2: tv.setText("下载完成"); break; } } }; final ImageButton button = (ImageButton)this.findViewById(R.id.button1); button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { new Thread(new Runnable() { @Override public void run() { int time = 10000; while(time > 0){ try { Thread.sleep(1000); time -= 1000; Message msg = new Message(); msg.what = 1; msg.arg1 = time; //也可以发送对象 //msg.obj = obj; handler.sendMessage(msg); } catch (InterruptedException e) { e.printStackTrace(); } } //下载完成,发送一个空消息 handler.sendEmptyMessage(2); } }).start(); } });