`
gybin
  • 浏览: 270270 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

android 简析自定义布局、布局的执行流程

 
阅读更多

代码示例针对Android 2.3

你玩过植物大战僵尸吗?你玩过愤怒的小鸟吗?你是不是很疑惑精美的UI界面是如何作出来的呢?很明显andriod 自带的控件是不可能做到那样的效果的,这里就用到了对控件、布局的重写。


单从重写控件来看,你会感觉到很简单(只需要覆盖onMeasure()onLayout()方法)就可以了,但是这两个方法的被谁调用?它的Framework层的布局流程究竟是怎样的,只有搞清楚这些我们才能很好的去重写布局,布上我们的View,从而实现我们想要的效果。

 

为了更直观地展示布局调用结构,我在这里简略绘制了布局Framework层的类图。

。。。。。。

请看此图

。。。。。。

 

View.java

// 注意final修饰,该方法永远不会被覆盖,整个布局结构 measure方法唯一

public final void measure(int widthMeasureSpec, int heightMeasureSpec) {

         onMeasure(widthMeasureSpec, heightMeasureSpec);

}

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

 

//注意final修饰,该方法永远不会被覆盖,整个布局结构layout方法唯一

public final void layout(int l, int t, int r, int b) {

        boolean changed = setFrame(l, t, r, b);

        if (changed || (mPrivateFlags & LAYOUT_REQUIRED) == LAYOUT_REQUIRED) {

                  onLayout(changed, l, t, r, b);

        }

}

protected void onLayout(boolean changed, int left, int top, int right, int bottom) { } 空方法


ViewGroup.java extends View.java 

  @Override

   protected abstract void onLayout(boolean changed, int l, int t, int r, int b);

// 测量该ViewGroup所包含的所有布局

protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {}

protected void measureChild(View child, int parentWidthMeasureSpec,

            int parentHeightMeasureSpec) {}

 

//我会单讲mChildren数组mChildren中的View如何来的。

public View getChildAt(int index) {  return mChildren[index];  }

public int getChildCount() {  return mChildrenCount; }

 

RelativeLayout.java extends ViewGroup.java

//当继承RelativeLayout布局时,我们应当覆盖该方法,以实现测量该布局包含的View//此处的实现并不能测量所有的View

 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

 

protected void onLayout(boolean changed, int l, int t, int r, int b) {}

private void measureChild(View child, LayoutParams params, int myWidth, int myHeight) {}

//还包含一个重要的内部类,代表RelativeLayout所包含的每一个view大小及位置信息

public static class LayoutParams extends ViewGroup.MarginLayoutParams{

          private int mLeft, mTop, mRight, mBottom;

}

 

下面我要自定义一个布局,定义布局的目的肯定是为了向其内部放置View

CustomGridLayout.java  extends RelativeLayout.java

初学者会问,我们到底需要继承RelativeLayout类的哪个方法呢!!

抛去一切,我们自己想象,布局控件需要

第一:控件(View)的大小 

第二:控件(View)的位置  

第三:知道要放置多少个View

 

通过熟读文档,我们应该知道:

onMeasure方法负责测量将要放在CustomGridLayout内部的View的大小。

onLayout方法负责分配尺寸及位置给将要放在CustomGridLayout内部的View

所以很明显,需要我们继承的方法是

1. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

功能:测量该布局所包含的所有View的大小(会在框架层循环取得每一个View,然后测量其大小),该方法会被View.java中的measure方法调用。而measure方法会被

2. protected void onLayout(boolean changed, int l, int t, int r, int b) {}

功能:在相应的位置放置相应的View,该方法会被View.java中的layout方法调用,而layout方法会被谁调用呢?

(1) 调用requestLayout()方法. 该方法内部会执行Object.layout(…)

(2) 直接调用 Object.layout(…) 

(3) 调用addView(View child, ...)时,

   调用addView(...)之前一般需要先调用android.view.View.setLayoutParams(LayoutParams params)


最后我简略分析了一下布局调用的流程调用,如下:

android <wbr>简析自定义布局、布局的执行流程

 也许有人会问当调用addView时,会和框架层的layout,onLayout,measure, onMeasure等几个布局方法有什么关系,或者说后者几个方法是怎么被触发的,别着急,看见addView(...)的底层实现了吗?

 

public void addView(View child, int index, LayoutParams params) {

        // addViewInner() will call child.requestLayout() when setting the new LayoutParams

        // therefore, we call requestLayout() on ourselves before, so that the child's 

        // request will be blocked at our level

        requestLayout(); 

        invalidate();

        addViewInner(child, index, params, false);

   }

很明显,addView在底层调用了requestLayout方法,该方法如时序图所示,会依次触发我们的onMeasure,onLayout方法。


  • 大小: 42.7 KB
分享到:
评论

相关推荐

    简析linux内核的内核执行流程.doc

    Linux内核的执行流程是操作系统启动过程的核心部分,它涉及到从硬件初始化到用户空间应用程序运行的各个阶段。这里我们将深入探讨Linux内核的启动过程,主要聚焦于从启动到`main.c`执行的关键步骤。 首先,Linux...

    android简析json数据

    本文将简析Android中处理JSON数据的几种方法,为Android开发者提供一些实用技巧。 首先,理解JSON的基本结构是至关重要的。JSON是一种轻量级的数据交换格式,它基于JavaScript的一个子集,采用完全独立于语言的文本...

    android 简析xml源码

    XML(eXtensible Markup Language)是一种用于标记数据的语言,广泛应用于Android系统中,用于存储配置文件、布局文件等。Android系统对XML的解析主要依赖于两个库:XmlPullParser和SAX(Simple API for XML)。这里...

    简析linux内核的内核执行流程.docx

    Linux内核的执行流程是操作系统启动过程中的关键环节,它涉及到计算机从硬件初始化到软件运行的整个转变。本文主要分析了Linux内核从启动到系统准备完毕的两个主要阶段。 第一阶段是从开机到`main.c`的执行。这一...

    简析linux内核的内核执行流程.rar

    理解Linux内核的执行流程对于系统开发、调试和优化至关重要。这篇文档将深入剖析Linux内核从启动到运行用户进程的整个过程。 一、引导加载器 Linux内核的执行流程始于引导加载器(Bootloader)。常见的引导加载器如...

    android descendantFocusability用法简析

    android descendantFocusability用法简析

    Android RelativeLayout相对布局属性简析

    RelativeLayout用到的一些重要的属性: 第一类:属性值为true或false android:layout_centerHrizontal 水平居中 android:layout_centerVertical 垂直居中 android:layout_centerInparent 相对于父元素完全居中 ...

    Android UI用户界面开发简析.pdf

    Android UI 用户界面开发简析 Android UI 用户界面开发是 Android 应用程序开发的重要组成部分,涉及到用户界面元素、布局实现、事件响应、应用软件代码目录结构等多个方面。下面将对这些方面进行详细的分析和说明...

    简析Android五大布局(LinearLayout、FrameLayout、RelativeLayout等)

    在Android开发中,布局是构建用户界面的关键组成部分。本文将深入解析Android的五大布局:LinearLayout、FrameLayout、RelativeLayout、AbsoluteLayout和TableLayout。 1. **LinearLayout(线性布局)** ...

    Android编程之线性布局LinearLayout实例简析

    线性布局(LinearLayout)是Android中最基础的布局方式之一,用于将子视图(Views)按照指定的方向(垂直或水平)进行排列。在XML布局文件中,我们可以通过`&lt;LinearLayout&gt;`标签来创建一个线性布局。以下是关于线性...

    Android PhoneGap简析

    PhoneGap的扩展性体现在其插件系统上,开发者可以创建自定义插件来扩展其功能,覆盖更多原生API。由于PhoneGap是开源的,社区活跃,有丰富的插件和文档支持,因此在成熟度上具有较高保障。 **结合地图应用的思考** ...

    NR PRACH信道和随机接入流程简析V2.0.pdf

    本文将探讨NR中的一个关键组成部分:PRACH(Physical Random Access Channel,物理随机接入信道)信道,及其相关的随机接入流程。 首先,PRACH信道是5G NR网络中的一个物理信道,用于无线设备(UE,User Equipment...

Global site tag (gtag.js) - Google Analytics