AppCompatActivity的setContentView()

    对于Android开发而言,Activity可以说是最重要也是最常用的组件了.每当我们创建一个Activity的时候,想要展示对应的视图,都需要调用setContentView(int layoutResId)来加载xml文件.对于setContentView()究竟发生了什么,我想大家都已经了解了很多.本质上是调用getWindow().setContentView(),而getWindow()又返回一个PhoneWindow.在PhoneWindow中installDecor()创建一个新的DecorView,根据不同的情况加载对应的布局.

    随着Android版本不断迭代更新,我们也慢慢的不去继承Activity了,而是选择AppCompatActivity,那么Activity和AppCompatActivity究竟有什么不同,我们可以从源码来了解一下.

    首先先来看一段代码

AppCompatActivity中的Activity.setContentVew()点击进入以后

    调用的方法还是同一个方法,但是调用的类已经由getWindow()变成了getDelegate().

    getDelegate()从字面上来理解,是获取一个委托或者代理,那么究竟代理了谁呢,让我们点击进去看一下.

点击getDelegate()

    这里我们可以看到,返回了一个通过AppCompatDelegate.create()创建的AppCompatDelegate对象.接下来继续点击进入看看这个AppCompatDelegate.create()究竟做了什么.

点击AppCompatDelegate.create()

    这里并没有做具体的逻辑处理,而是创建了一个AppCompatDelegateImpl(),根据Java的一些约定俗成的规范,看见Impl结尾的类一定会有想要的东西,我们就进入这个AppCompatDelegateImpl找一下有没有setContentView().

进入一看,果然有setContentView()

    既然找到了我们想要的setContentView,那么我们就来具体分析一下这几个方法都干了什么.

    1.首先调用了ensureSubDecor();

    2. mSubDecor就是一个ViewGroup,内部包含了一个id为content的ViewGroup,把我们自己的布局放入contentParent中.

    3.这样来看,最关键的方法就在ensureSubDecor()中了.


    再进入ensureSubDecor()这个方法之前,我们先稍微捋顺一下整体的流程

AppCompatActivity.setContentView()流程

   前期准备结束,我们开始进入这个ensureSubDecor();

ensureSubDecor()

    1.首先显示一个if判断,这个mSubDecorInstalled默认为false

    2.mSubDecor = createSubDecor(),这里是创建出加载布局的ViewGroup

    3.onSubDecorInstalled(mSubDecor),是SubDecor创建完之后的回调

    4.将mSubDecorInstalled变为true.


    createSubDecor()

这里可以看出,如果使用AppCompatActivity不设置主题,那么就会报错
接上面的

    这里我们看到,首先是初始化一些特征的标识.mWindow其实就是Window,那么这个Window如何创建,我们先放一放.之后可以看见,之前的subDecorView就是一个ViewGroup.

    因为我们默认是NOTITLE的,所以我隐藏掉了一些方法,直接看主要的.mOverlayActionMode其实是区分是否调用了requestWindowFeature().requestWindowFeature()内部其实就是设置Activity全屏.我们从这里知道,为什么需要在setContentView()之前调用这个方法设置Activity全屏.

    如果我们什么都不设置,就会走else方法.映射出来一个subDecor.这个subDecor的布局究竟是什么,我们也是暂时放下.

接上面

    再往下看,我们看到,通过subDecor找到了一个ContentFrameLayout contentView.也就是一个FrameLayout.

    在Activity.setCountView()中,我们可以知道,R.id.content是父容器的id.这里contentView的id被赋值为R.id.content.那么我我们姑且认为,contentView是我们的父容器.

    最后mWindow.setContent(subDecor),Windows还是我们之前的PhoneWindow,并且把subDecor添加进入.

    我们回到上边,先找到 R.layout.abc_screen_simple,看看subDecor是个什么东西.

    xml的代码我就不贴了,subDecor最外层是一个FitWindowsLinearLayout,里面放置了一个ActionBar以及ContentFrameLayout.ContentFrameLayout就是subDecor找到的contentView,也就是设置了R.id.content的contentVIew.

    大致布局入下图

subDecor布局

    流程走到这里,基本就明朗化了,还剩两个最重要的东西.一个是mWindow.getDecor(),一个是mWindow.setContentView(subDecor).


     首先我们来看mWindow.getDecor()究竟是什么.

    当我们点击进入Window源码的时候会看见这一行注释

Window.java

    注意<P>里面的一段话,就是说Window是一个抽象类,它的实现类是PhoneWindow.那么我就去找PhoneWindow

PhoneWindow.getDecorView()

    PhoneWindow当中找到了getDecorView()方法,可以看出返回的是一个View.如果这个View为null,就去创建这个View.我们继续进入installDecor()

installDecor()重点是两个框住的方法

    在installDecor()中,如果是第一次调用,那么就先generateDecor(),之后有调用generateLayout(),把mDecor传入了进去.

generateDecor()

    generateDecor()其实就是创建了一个DecorView,而这个DecorView其实就是继承FrameLayout.

    generateLayout()内部主要根据不同主题设置了一些主题资源,并且找到了一个类似subDecor的布局加入到了DecorView中,最后返回了一个ContentParent. 


    到现在为止,mWindow.getDecorView()我们捋顺的差不多了.

    1.创建了DecorView,并且根据不同的主题添加到DecorView中.

    2.找到R,id,content, 也就是mContentParent,

    现在我们来看最后一个方法,mWindow.setContentView(subDecor)

setContentView(subDecor)

    setContentView也在PhoneWindow中实现了.

    1.mContentParent已经在getDecorView()中创建了,所以不会为null

    2.之后会判断有没有transitions动画,默认没有动画,那么进入else

    3.调用addView方法将view也就是subDecor添加到了mContentParent中


    那么整个createSubDecor()流程就是这样.  

    1.mWindow.getDecorView();创建了DecorView以及mContentParent

    2.mWindow.setContentView(subDecor);把subDecor放到了mContentParent里面


    那么,我们在回头看一眼AppCompatDelegateImpl.setContentView();

AppCompatDelegateImpl.setContentView()

    1.创建DecorView 和 subDecor

    2.调用createSubDecor的时候把原本是R.id.content的windowContentView设置成了NO_ID,并且将contentView也就是ContentFrameLayout设置成了R.id.content,此时的contentParent就是ContentFrameLayout.

    3.将布局放入contentParent中.

    4.将我们的布局id映射成View并且放到contentParent下


    最后总结一下AppCompatActivity.setContentView()

    1.当我们调用setContentView的时候,加载了两次系统布局,一次找到contentView,设置id为R.id.content.一次找到了windowContentView.设置id为NO_ID;

    2.在PhoneWindos中创建了DecorView,是最底层的View,最后将布局放入到ContentFrameLayout中.

    最后梳理一下简单的流程图

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352