关于非主线程中操作ui的实现

来源:互联网 发布:ios手游模拟器 mac 编辑:程序博客网 时间:2024/05/22 04:02

Handler可以看作是一个工具类,通常与looper进行交互,即向消息队列中插入消息。

Looper用来为一个线程开启一个消息循环。

通常Handler会与被定义时所在线程的looper绑定。new Handler()时实际是new Handler(Looper.myLooper()),即获取当前线程的looper对象

大家都知道在非主线程定义Handler会报错,那是因为在非主线程中没有自动开启一个消息循环,而handler发送消息时是要发送到looper所管理的消息队列中的,所以会报错。而在主线程不报错是因为系统自动在主线程中创建了一个Looper。

所以在子线程中也可以使用handler,前提是new handler之前需要创建looper。其实android中有一个类:HandlerThread,创建一个HandlerThread,即创建了一个包含Looper的线程。然后可以创建handler,new Handler(mHandlerThread.getLooper()).就可以实现在子线程中使用handler处理消息。

由此我想到了在子线程中弹toast。不作处理的话是无法顺利实现弹toast的。原因是:toast显示是通过new Handler之后post给消息队列的,最后通过windowManager的add view显示出来。子线程没有创建looper,当然也就无法创建handler。解决方法大家都知道:在使用toast之前Looper.prepare(),这句是创建一个Looper,最后通过looper.loop来取消息队列中的消息来作处理。

那么是不是就可以在非主线程中操作view了呢?请看下文:

android中view的操作最终都会调用到viewRoot这个类的invalidateChild与requestLayout这两个接口,而这两个接口又都调用了

checkThread这个方法:

void checkThread(){

if (mThread != Thread.currentThread()){

thrownew CalledFromWrongThreadException(

"Only the original thread that created a view hierarchy can touch its views.");

}

}

所以,如果调用当前接口的线程与ViewRoot中维护的线程mThread不同的话就会报错。其实想要在非主线程操作view,就需要在非主线程中构造ViewRoot。而mThread赋值也正是在ViewRoot中进行的。下面请看:

publicfinalclassViewRootextends HandlerimplementsViewParent, View.AttachInfo.Callbacks {  

publicViewRoot(Context context) {  

super();  

......  

mThread = Thread.currentThread();  

......  

}

}

Activity与dialog都是通过Windowmanager的add view来通知系统要显示的窗口。实现add view方法的是windowmanagerimpl类

publicclassWindowManagerImplimplements WindowManager {

........  

privatevoidaddView(View view, ViewGroup.LayoutParams params, boolean nest)  

{  

if (Config.LOGV) Log.v("WindowManager","addView view=" + view);  

if (!(params instanceof WindowManager.LayoutParams)) 

{  

thrownew IllegalArgumentException("Params must be WindowManager.LayoutParams");}  

final WindowManager.LayoutParams wparams = (WindowManager.LayoutParams)params;  

ViewRoot root;  

View panelParentView =null;  

synchronized (this) {  

........  

root =new ViewRoot(view.getContext());  

root.mAddNesting =1;  

........  

}  

// do this last because it fires off messages to start doing things  

root.setView(view, wparams, panelParentView);  

}  

........

}

可见添加addview的时候会构造一个ViewRoot,

而对于activity add view是封装在setcontentview中的,但是这个方法是在oncreate中调用的,increase又是运行在主线程的。

但对于dialog则完全可以在子线程中去构造,也就可以通过处理在子线程中运行。


0 0
原创粉丝点击