博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Activity的显示之Window和View
阅读量:6328 次
发布时间:2019-06-22

本文共 5236 字,大约阅读时间需要 17 分钟。

该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,如果能给各位看官带来一丝启发或者帮助,那真是极好的。


前言

还是先来个最简单的HelloWord代码,用Android Studio 3.0新建项目(一直使用默认)后会自动生成一个HelloWorld的项目,如下:

activity_main.xml

MainActivity.java

package com.helloword;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;public class MainActivity extends AppCompatActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);    }}

Window初探

我们启动应用程序,安装到手机或者模拟器上,界面显示了一行字“Hello World!”。我们的激动无以言表。下面来看看MainActivity是如何显示的。(这次不看生命周期了)

有些同学看到这里或许有些迷惑,Android Studio IDE自动生成了activity_main.xml和MainActivity,然后我运行到手机或者模拟器上,,它就这样显示出来了,,要问为什么?我还真没想过。。。好吧,让我稍微撩起一点Activity神秘的面纱。

WindowManager

在此之前我们要先猜想一波,Activity是以什么形式显示的呢?我们知道Windows有Window(窗口)这个概念,其实在Android中也有这个概念,与Windows的窗口有着类似的含义。Android中的所有视图,不管是Activity,Dialog,还是Toast都是附加在Window上展示的。介绍了Window,开始介绍如何操作Window(Window是个抽象类)。WindowManager是操作Window的入口。

WindowManager继承自ViewManager的函数

public void addView(View view, ViewGroup.LayoutParams params);public void updateViewLayout(View view, ViewGroup.LayoutParams params);public void removeView(View view);

都是顾名思义的函数命名,增加、更新、删除。可以看到WindowManager操作的主要对象为View.那么我们先看一下View。在此之前依然的先上代码。

activity_main.xml(没变哦)

MainActivity.java(变了哦)

package com.helloword;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;public class MainActivity extends AppCompatActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);                /**         * 可以看到我们熟悉的setContentView(R.layout.activity_main)调用没有了,         *而是采用了下面的代码,运行然后居然得到了和setContentView(R.layout.activity_main)         *的一样的显示结果,难道说setContentView(R.layout.activity_main)         * 逻辑这么简单吗?我的回答是否定的,setContentView(R.layout.activity_main)         *逻辑并没有那么简单,不信的话你点击返回按钮试试看。(无法操作了哦,         *不是你们的手机是不是这样,反正我的手机点击返回没有作用,重写onBackPressed方法也没有调用)         *          * (注:我以下面这种方式得到了和setContentView(R.layout.activity_main)一样的显示效果         *,可这只是为了分析方便,不要模仿哦    )         */                 //① 解析.activity_main.xml文件并创建View并且指定其父View为null(即没有父View)        View view = getLayoutInflater().inflate(R.layout.activity_main,null);        //② 设置显示参数        WindowManager.LayoutParams params = new WindowManager.LayoutParams(                WindowManager.LayoutParams.MATCH_PARENT,                WindowManager.LayoutParams.MATCH_PARENT,                1,                0,                PixelFormat.TRANSPARENT);        //③ 添加View        getWindowManager().addView(view,params);    }}

不管如何我们使用上面的代码获得了和setContentView(R.layout.activity_main)一样的显示结果,setContentView(R.layout.activity_main)的源码我们不具体分析,我不想把你们的思维带进如汪洋大海般的源码中而无法自拔。那我们来看一下上面的代码中的含义,基本上注释里都写了。不过我们还是来详细解释一下,以序号为顺序。

解析activity_main.xml文件并创建View并且指定其父View为null(即没有父View)

inflate函数声明如下public View inflate(@LayoutRes int resource, @Nullable ViewGroup root)

我们传入的是activity_main.xml的资源ID以及null,即函数内部会解析XML文件并最终创建View,

后面参数传null的目的是我们不应该为该View指定父View,因为下文中会指定。

设置显示参数

WindowManager.LayoutParams的构造函数比较多,我们使用的是

**LayoutParams(int w, int h, int _type, int _flags, int _format)** 为即将显示的Window指定显示参数,我们传入的是            WindowManager.LayoutParams.MATCH_PARENT,            WindowManager.LayoutParams.MATCH_PARENT,            1,            0,            PixelFormat.TRANSPARENT

w(宽)h(高)我们比较好理解,我们来看一下_type、_flags、_format

注:上面两图出自《Android开发艺术探索》

添加View

看到了很多博客都是从setContentView(R.layout.activity_main)一路分析,结合源码,各位大神都讲的非常清晰。确实我们绝大部分应用都是使用这种方法,包括我,我也十分不推荐上面我的那种写法。上面也说了那样做是为了更好的分析。

setContentView(R.layout.activity_main)与我自己写的代码的对比

先来看一下setContentView(R.layout.activity_main)

先来一张Android视图模型图,接着是setContentView(R.layout.activity_main)调用流程图

我们来看一下启动的View层级图

再来看一下我们的MainActivity代码,没有使用setContentView(R.layout.activity_main)哦

流程图如下

其层级图如下

setContentView(虽然这个网上一搜到处都是,不过我的不一样哦)

还是先上代码

activity_main.xml(没变哦)

MainActivity.java(变了哦)

package com.helloword;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;public class MainActivity extends AppCompatActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        //注意我们先注释掉该句        //setContentView(R.layout.activity_main);    }}

不是要分析setContentView(R.layout.activity_main)为什么注释掉呢,,先别急,我们先运行一下,居然没有崩溃,只是得到一个空白内容页面(这个显示跟Activity Theme主题设置相关,我的是默认Theme)如下

我们来看一下View层级图

需要注意的地方我用红色框框圈住了,可以看到我们在没有写setContentView(R.layout.activity_main)的时候,程序依然可以正常运行,并显示出Title。而且结合上文中

这个图的话,我们发现除了红色区域没有,其他区域的都有。那么该Activity的显示的基础——PhoneWindow和顶级View——DecorView是什么时候创建的呢?

其实上文中已经提到了一些,我们就提到了Activity的创建工作是在android.app.ActivityThread的main函数中完成的。那么我们来分析一下该函数(我们不具体分析源码,太多了),不过我会上图

好了,接着把//setContentView(R.layout.activity_main);//去掉。又得到我们熟悉的HelloWorld界面,

这次我们再来看一下这个层级图是不是理解的更清晰了呢。我们的setContentView(R.layout.activity_main);内部把activity_main.xml文件解析成View,并把该View添加到contentView

注:调用setContentView(R.layout.activity_main)和不调用生成DecorView的逻辑是不一样的,如果调用了该函数会在该函数内部判断生成DecorView,如果不调用的话会在handleResumeActivity的时候判断生成这就是我们不写setContentView(R.layout.activity_main),也会生成DecorView的原因


本篇总结

Android系统控件本身帮助我们做了很多事情,我们使用setContentView(R.layout.activity_main)之前便得到了一个PhoneWindow以及DecorView,并通过该函数把我们自己的布局View放到了DecorView下。理解这一点对后面的文章阅读至关重要。


下篇预告


参考博文


此致,敬礼

转载地址:http://ynwoa.baihongyu.com/

你可能感兴趣的文章
Linux系统权限
查看>>
TinyTemplate模板引擎火热出炉,正式开源了~~~
查看>>
android开发之GPS定位详解
查看>>
Mac OS X如何重装 苹果电脑重装操作系统
查看>>
集算器读写EXCEL文件的代码示例
查看>>
Ubuntu Server上搭建可用于生产环境的ASP.NET服务器
查看>>
php---PHP使用GD库实现截屏
查看>>
华为交换机802.1x动态下发vlan配置
查看>>
spring boot websocket + thy模版
查看>>
查看文件的真实路径
查看>>
如何开发一个自己的 RubyGem?
查看>>
centos 7 修改主机名的方法
查看>>
WSUS系列之二:WSUS角色安装
查看>>
大数据作业第4天
查看>>
职工系统150206308
查看>>
『中级篇』K8S最小调度单位Pod(62)
查看>>
我的友情链接
查看>>
ACE网络编程思考(一)
查看>>
数据结构的几种存储方式
查看>>
React源码学习——ReactClass
查看>>