Android开发之漫漫长途 Ⅱ——Activity的显示之Window和View(1)

该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽可能按照先易后难的顺序进行编写该系列。该系列引用了《Android开发艺术探索》以及《深刻理解Android 卷Ⅰ,Ⅱ》中的相关知识,另外也借鉴了其余的优质博客,在此向各位大神表示感谢,膜拜!!!另外,本系列文章知识可能须要有必定Android开发基础和项目经验的同窗才能更好理解,也就是说该系列文章面向的是Android中高级开发工程师。java

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

activity_main.xmlapp

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.helloword.MainActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</android.support.constraint.ConstraintLayout>

MainActivity.javaide

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);
    }
}

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

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


在此以前咱们要先猜测一波,Activity是以什么形式显示的呢?咱们知道Windows有Window(窗口)这个概念,其实在Android中也有这个概念,与Windows的窗口有着相似的含义。Android中的全部视图,不论是Activity,Dialog,仍是Toast都是附加在Window上展现的。介绍了Window,开始介绍如何操做Window(Window是个抽象类)。WindowManager是操做Window的入口。 WindowManager继承自ViewManager的函数code

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(没变哦)orm

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.helloword.MainActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</android.support.constraint.ConstraintLayout>

MainActivity.java(变了哦)xml

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) 先来一张Android视图模型图,接着是**setContentView(R.layout.activity_main)**调用流程图

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

再来看一下咱们的MainActivity代码,没有使用setContentView(R.layout.activity_main)哦 流程图以下

其层级图以下


很差意思时间不太够了,,具体的分析明天继续Android开发之漫漫长途 Ⅱ——Activity的显示之Window和View(2)

相关文章
相关标签/搜索