Android都需要在主线程刷新UI吗?(以下环境在android5.1中分析)

在子线程刷新UI一般会有这样的异常:CalledFromWrongThreadException
该异常来自于framework/base/core/java/android/view/ViewRootImpl.java中的
void checkThread() {
if (mThread != Thread.currentThread()) {
throw new CalledFromWrongThreadException(
“Only the original thread that created a view hierarchy can touch its views.”);
}
}
其中在ViewRootImpl的构造中会发现:
public

ViewRootImpl(Context context, Display display) {
mContext = context;

mWindowSession = WindowManagerGlobal.getWindowSession();mDisplay = display;mBasePackageName = context.getBasePackageName();mDisplayAdjustments = display.getDisplayAdjustments();mThread = Thread.currentThread();...

}
而每次在更新View的时候都会调用checkThread方法来检测线程情况,因此如果需要在子线程中刷新UI的话则需要在子线程中创建ViewRoot(5.1中是ViewRootImpl)了,但是ViewRootImpl并不能直接new,它会在哪里创建呢?我们可以看到类:android.view.WindowManagerImpl该
类实现 WindowManager,其中在addView的时候如下:
public void

addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
applyDefaultToken(params);
mGlobal.addView(view, params, mDisplay, mParentWindow);
}
进入WindowManagerGlobal找到addView方法:发现这么个关键地方
public void addView(View view, ViewGroup.LayoutParams params,Display display, Window parentWindow) {

ViewRootImpl root;

root = new ViewRootImpl(view.getContext(), display);
view.setLayoutParams(wparams);
mViews.add(view);
mRoots.add(root);
mParams.add(wparams);
}
到此我们知道了他们关联的地方,那么如果想要在子线程更新的话也就是只需要在子线程addView就可以了,测试例子如下:
public class MainActivity extends Activity implements OnClickListener
{

@Overrideprotected void onCreate(Bundle savedInstanceState){    // TODO Auto-generated method stub    super.onCreate(savedInstanceState);    setContentView(R.layout.activity_main);    findViewById(R.id.m_btn_show).setOnClickListener(this);    findViewById(R.id.m_btn_hide).setOnClickListener(this);}@Overridepublic void onClick(View v){    // TODO Auto-generated method stub    switch (v.getId())    {        case R.id.m_btn_show:            test();            break;        case R.id.m_btn_hide:            mHandler.sendEmptyMessage(1);            break;    }}private Handler mHandler;private void test(){    new Thread()    {        @Override        public void run()        {            // TODO Auto-generated method stub            super.run();            Looper.prepare();            final TextView tx = new TextView(MainActivity.this);            tx.setText("test11111111111111111");            final WindowManager wm = MainActivity.this.getWindowManager();

// WindowManager.LayoutParams params =
// new WindowManager.LayoutParams(250, 250, 200, 200, WindowManager.LayoutParams.FIRST_SUB_WINDOW, WindowManager.LayoutParams.TYPE_TOAST, PixelFormat.OPAQUE);
WindowManager.LayoutParams params = new WindowManager.LayoutParams();
params.flags = LayoutParams.FLAG_NOT_FOCUSABLE|LayoutParams.FLAG_NOT_TOUCHABLE;
params.width = 200;
params.height = 200;
params.x = 100;
params.y = 100;
params.type = LayoutParams.FIRST_SUB_WINDOW;
wm.addView(tx, params);
mHandler = new Handler()
{

                @Override                public void handleMessage(Message msg)                {                    // TODO Auto-generated method stub                    super.handleMessage(msg);                    wm.removeView(tx);                }            };            Looper.loop();        }    }.start();}

}
从这里也可以看出Android视图和线程的关系。
问题:为什么addView过程(也就是new ViewRootImpl)的时候要放在Looper.prepare()和Looper.loop()之间?
因为在ViewRootImpl的构造中有创建Handler:
mAttachInfo = new View.AttachInfo(mWindowSession, mWindow, display, this, mHandler, this);

你可能会注意到这里的addView中有
Display,这是干嘛的?查看以下引用发现如下:

其实它表示显示的载体(可以在多屏幕情况下使用)

更多相关文章

  1. android 动态控制状态栏显示和隐藏的方法实例
  2. android之ListView的item布局问题总结
  3. Android中ExpandableListView控件基本使用
  4. Android(安卓)studio 3.1.4 无法获取pom文件
  5. 移动端H5的Video标签无法播放在线视频的问题
  6. 安卓开发入门学习笔记
  7. 上官网学android之二(Building your First APP)
  8. Android线程调度机制
  9. Android与J2ME区别之我见(1)

随机推荐

  1. golang核心原理-协程栈
  2. 数据不撒谎,Flink-Kafka性能压测全记录!
  3. golang核心原理-协程调度时机
  4. 《MySQL》系列 - 十张图详解 MySQL 日志(
  5. 【腾讯二面】5s内建立多少个mysql连接?
  6. Java异常与调优一站式解决方案
  7. 昨天面试别人说他熟悉Flink,结果我问了他F
  8. js作业一常识类
  9. 2021-04-04:给定一个非负数组arr,和一个正
  10. 2021必修 首门CSS架构系统精讲 理论+实战