setContentView大家都不陌生,他是我们平常写页面必须要实现的方法,也不废话,直接上图上代码,本文讲分Activity和AppCompatActivity来讲解setContentView
1.首先直接看Activity的setContentView方法
data:image/s3,"s3://crabby-images/18ac3/18ac3246cee7475ec1383af50a58070fce387083" alt=""
通过window来调用setContentView的方法,那么window又是什么呢? 这个window其实是一个phoneWindow,在activity的attach创建了window=new PhoneWindow(),至于attach什么调用的后面给大家分析.居然是PhoneWindow,那么直接看PhoneWindow的setContentView方法
1.PhoneWindow.setContentView
data:image/s3,"s3://crabby-images/a3acc/a3accdca9a0ce4745b8b30f503a3a60632548038" alt=""
首先第一步会调用installDecor方法,因为mDecor是null,所以会调用generateDecor方法创建一个mDecor,然后调用generateLayout方法,generateLayout方法会根据标记去设置各种各样的title,然后创建一个view赋值给generateLayout
data:image/s3,"s3://crabby-images/f7989/f79892f6affd55b9eafef68f66d0bde94084fb0d" alt=""
generateLayout代码比较长,就拿一个简单说一下,首先是根据设置的标记创建相应的布局,例如R.layout.screen_simple布局然后onResourcesLoaded方法它是将R.layout.screen_simple布局渲染到我们的Decor上面去.,然后拿到id, ID_ANDROID_CONTENT==com.android.internal.R.id.content,再然后通过mLayoutInflater.inflate(layoutResID, mContentParent);渲染我们自己的写的布局
data:image/s3,"s3://crabby-images/52fa4/52fa4419c540723fbda8837ea1f2dc1e14ce8bf7" alt=""
data:image/s3,"s3://crabby-images/3dee2/3dee227c6143ed2e753da15838bc9d05302b0008" alt=""
总结一下:activity里面创建了一个window,window里面创建了一个DecorView,DecorView里面有个xml布局,这个xml布局有个FrameLayout 他的id是content,通过获取id+content将我们自己activity_main布局添加到这个FrameLayout,最后将xml布局添加到DecorView上面去
data:image/s3,"s3://crabby-images/64138/64138d474b7afc68ceb18a2ef7b8d78d3b40d5f0" alt=""
2.AppCompatActivity的setContentView方法
data:image/s3,"s3://crabby-images/df0c7/df0c7143b785de44387f651cab17aeb18f3967b3" alt=""
getDelegate()调用的是AppCompatDelegateImpl,所以直接看AppCompatDelegateImpl的setContentView方法.首先第一步先分析ensureSubDecor,ensureSubDecor先会调用createSubDecor方法
data:image/s3,"s3://crabby-images/9eb96/9eb96b80ed97f3045ae67af0f69fb4317d1688c4" alt=""
data:image/s3,"s3://crabby-images/9e58d/9e58dfc82717c99bf05cea44e563c0ca66bc7093" alt=""
createSubDecor首先首先会根据设置加载不同的主题样式,然后调用ensureWindow,需要注意的是这里调用ensureWindow的目的只是判空的,因为在onCreate的时候就已经调用了ensureWindow方法创建了mWindow,然后通过mWindow,拿到DecorView. mWindow就是PhoneWindow
data:image/s3,"s3://crabby-images/83ef5/83ef5b0a37dfe73fd00b2f73f1c5080e2f9aa17b" alt=""
data:image/s3,"s3://crabby-images/00980/00980d143794a21cca8f9d4b6c141c9c6a1b8922" alt=""
然后再回到createSubDecor方法,根据不同状态设置相应的subDecor,例如加载abc_screen_simple布局
data:image/s3,"s3://crabby-images/d0f6b/d0f6bc931bc72a6cbdcc2141e2e55d42e9e2672c" alt=""
data:image/s3,"s3://crabby-images/751ff/751ff4acb892b3ee8606d4db42bd330b887c2597" alt=""
data:image/s3,"s3://crabby-images/2de48/2de48416d677326a2edb62a2052c266140482ee4" alt=""
加载布局之后呢,通过获取id拿到这个ContentFrameLayout,然后又通过mWindow拿到ViewGorup,这里的mWindow拿到的就是上面的R.layout.screen_simple布局的id,然后进行循环移除screen_simple里面的子view,并且把移除的这个view添加的到abc_screen_simple里面去,然后将ViewGorup的id设置为空,然后设置contentView的id
可以这么理解原始的R.layout.screen_simple里面有一个id.content,然后我把它的id设置为空,然后将R.layout.abc_screen_simple的里面的ID设置为id.content,也就是相当于在R.layout.screen_simple又包裹了abc_screen_simple,接着调用mWindow.setContentView(subDecor);
data:image/s3,"s3://crabby-images/9a965/9a9651e2daa85a86b10966cf419e5b071dd3902f" alt=""
通过上述的流程就拿到了mContentParent了,然后调用了PhoneWindow.setContentView(view)通过addView添加到mContentParent里面,接着回到就执行了LayoutInflater.from(mContext).inflate(resId, contentParent),进行布局的填充
data:image/s3,"s3://crabby-images/f3df1/f3df1d23168cd3334d923ae6311f1bdb6fb37d8e" alt=""
总结一下:大致流程和activity里面的流程差不多,也是拿到window,window里面创建了一个DecorView,DecorView里面有个abc_screen_simple.xml布局,这个abc_screen_simple.xml布局有个ContentFrameLayout 他的初始id是action_bar_activity_content,然后在获取R.layout.screen_simple的id.content,先是遍历移除screen_simple的子view,然后将移除的view添加abc_screen_simple里面去,最后讲screen_simple的id设置为空,把screen_simple的id设置给abc_screen_simple,这里主要是兼容性,所以将id设置为screen_simple的id,最后是LayoutInflater.from(mContext).inflate(resId, contentParent),进行布局的填充
data:image/s3,"s3://crabby-images/49a55/49a55ef178aa7d23cc6f38148da388832af711b5" alt=""
3.LayoutInflater.from(mContext).inflate(resId, contentParent)
会先进行xml的解析,然后调用inflate方法,在inflate里面调用createViewFromTag方法,这个方法注意目的就是创建rootView
data:image/s3,"s3://crabby-images/16ccf/16ccf742a8eb95bf79d9ce6b6a6089bffbcfd9e8" alt=""
LayoutInflater.createViewFromTag方法怎么创建rootView的呢?首先通过判断是否包含indexOf判断来创建不同的view,其目的就是区分是否是自定义view,然后将其包名补全.举个例子说明下androidx.constraintlayout.widget.ConstraintLayout进入了if了
data:image/s3,"s3://crabby-images/5a7d6/5a7d6a7a70c4cc3fc58c51acd373ab8782890955" alt=""
经过周转最后调用了PhoneLayoutInflater的onCreateView方法,它for循环调用了createView方法
data:image/s3,"s3://crabby-images/b8ae4/b8ae4ae50fe76958d904c08616fdc5ec883e79bf" alt=""
通过反射创建view对象
data:image/s3,"s3://crabby-images/85ca9/85ca922d412829978804e0432f327eaaa8b081d9" alt=""
4.创建子View
viewRoot创建之后后,我们还需要创建子view,我们回到inflate方法里面的rInflate方法,在rlnflate里面根据不同的类型执行相应的操作,为了方便我们直接看createViewFromTag方法,这个方法是不是很熟悉,之前调用过了,创建createViewFromTag之后设置参数,然后通过addView添加到viewGroup里面去
data:image/s3,"s3://crabby-images/adbd8/adbd82527272326390b2f18e33c8d959e5dcac0d" alt=""
setContentView分析到这里就大致分析完了,有问题提出问题改正
网友评论