专栏名称: 安卓开发精选
伯乐在线旗下账号,分享安卓应用相关内容,包括:安卓应用开发、设计和动态等。
目录
相关文章推荐
51好读  ›  专栏  ›  安卓开发精选

如何优化你的布局层级结构之RelativeLayout和LinearLayout及FrameLayout性能分析(下)

安卓开发精选  · 公众号  · android  · 2016-09-14 08:32

正文

请到「今天看啥」查看全文


(点击 上方公众号 ,可快速关注)


来源:伯乐在线专栏作者 - yuiop逆流的鱼

链接:http://android.jobbole.com/84631/

点击 → 了解如何加入专栏作者


接上文


看下FrameLayout的源码,做了什么?


protected void onMeasure ( int widthMeasureSpec , int heightMeasureSpec ) {

int count = getChildCount ();

final boolean measureMatchParentChildren =

MeasureSpec . getMode ( widthMeasureSpec ) != MeasureSpec . EXACTLY ||

MeasureSpec . getMode ( heightMeasureSpec ) != MeasureSpec . EXACTLY ;

//当FrameLayout的宽和高,只有同时设置为match_parent或者指定的size,那么这个

//measureMatchParentChlidren = false,否则为true。下面会用到这个变量

mMatchParentChildren . clear ();

int maxHeight = 0 ;

int maxWidth = 0 ;

int childState = 0 ; //宽高的期望类型

for ( int i = 0 ; i count ; i ++ ) { //一次遍历每一个不为GONE的子view

final View child = getChildAt ( i );

if ( mMeasureAllChildren || child . getVisibility () != GONE ) {

//去掉FrameLayout的左右padding,子view的左右margin,这时候,再去

//计算子view的期望的值

measureChildWithMargins ( child , widthMeasureSpec , 0 , heightMeasureSpec , 0 );

final LayoutParams lp = ( LayoutParams ) child . getLayoutParams ();

/*maxWidth找到子View中最大的宽,高同理,为什么要找到他,因为在这里,FrameLayout是wrap

-content.他的宽高肯定受子view的影响*/

maxWidth = Math . max ( maxWidth ,

child . getMeasuredWidth () + lp . leftMargin + lp . rightMargin );

maxHeight = Math . max ( maxHeight ,

child . getMeasuredHeight () + lp . topMargin + lp . bottomMargin );

childState = combineMeasuredStates ( childState , child . getMeasuredState ());

/*下面的判断,只有上面的FragLayout的width和height都设置为match_parent 才不会执行

此处的mMatchParentChlidren的list里存的是设置为match_parent的子view。

结合上面两句话的意思,当FrameLayout设置为wrap_content,这时候要把所有宽高设置为

match_parent的子View都记录下来,记录下来干什么呢?

这时候FrameLayout的宽高同时受子View的影响*/

if ( measureMatchParentChildren ) {

if ( lp . width == LayoutParams . MATCH_PARENT ||

lp . height == LayoutParams . MATCH_PARENT ) {

mMatchParentChildren . add ( child );

}

}

}

}

// Account for padding too

maxWidth += getPaddingLeftWithForeground () + getPaddingRightWithForeground ();

maxHeight += getPaddingTopWithForeground () + getPaddingBottomWithForeground ();

// Check against our minimum height and width

maxHeight = Math . max ( maxHeight , getSuggestedMinimumHeight ());

maxWidth = Math . max ( maxWidth , getSuggestedMinimumWidth ());

// Check against our foreground's minimum height and width

final Drawable drawable = getForeground ();

if ( drawable != null ) {

maxHeight = Math . max ( maxHeight , drawable . getMinimumHeight ());

maxWidth = Math . max ( maxWidth , drawable . getMinimumWidth ());

}

//设置测量过的宽高

setMeasuredDimension ( resolveSizeAndState ( maxWidth , widthMeasureSpec , childState ),

resolveSizeAndState ( maxHeight , heightMeasureSpec ,

childState MEASURED_HEIGHT_STATE_SHIFT ));

count = mMatchParentChildren . size (); //这个大小就是子view中设定为match_parent的个数

if ( count > 1 ) {

for ( int i = 0 ; i count ; i ++ ) {

//这里看上去重新计算了一遍

final View child = mMatchParentChildren . get ( i );

final MarginLayoutParams lp = ( MarginLayoutParams ) child . getLayoutParams ();

int childWidthMeasureSpec ;

int childHeightMeasureSpec ;

/*如果子view的宽是match_parent,则宽度期望值是总宽度-padding-margin

如果子view的宽是指定的比如100dp,则宽度期望值是padding+margin+width

这个很容易理解,下面的高同理*/

if ( lp . width == LayoutParams . MATCH_PARENT ) {

childWidthMeasureSpec = MeasureSpec . makeMeasureSpec ( getMeasuredWidth () -

getPaddingLeftWithForeground () - getPaddingRightWithForeground () -

lp . leftMargin - lp . rightMargin ,

MeasureSpec . EXACTLY );

} else {

childWidthMeasureSpec = getChildMeasureSpec ( widthMeasureSpec ,

getPaddingLeftWithForeground () + getPaddingRightWithForeground () +

lp . leftMargin + lp . rightMargin ,

lp . width );

}

if ( lp . height == LayoutParams . MATCH_PARENT ) {

childHeightMeasureSpec = MeasureSpec . makeMeasureSpec ( getMeasuredHeight () -

getPaddingTopWithForeground () - getPaddingBottomWithForeground () -

lp . topMargin - lp . bottomMargin ,

MeasureSpec . EXACTLY );

} else {

childHeightMeasureSpec = getChildMeasureSpec ( heightMeasureSpec ,

getPaddingTopWithForeground () + getPaddingBottomWithForeground () +

lp . topMargin + lp . bottomMargin ,

lp . height );

}

//把这部分子view重新计算大小

child . measure ( childWidthMeasureSpec , childHeightMeasureSpec );

}

}

}



加了一个嵌套,onMeasure时间,多了将近一倍,原因在于:LinearLayout在某一方向onMeasure,发现还存在LinearLayout。将触发


if ( useLargestChild && ( heightMode == MeasureSpec . AT_MOST || heightMode == MeasureSpec . UNSPECIFIED )) {

mTotalLength = 0 ;

for ( int i = 0 ; i count ; ++ i ) {

final View child = getVirtualChildAt ( i );

if ( child == null ) {

mTotalLength += measureNullChild ( i );

continue ;

}

if ( child . getVisibility () == GONE ) {

i += getChildrenSkipCount ( child , i );

continue ;

}

}


因为二级LinearLayout父类是Match_parent,所以就存在再层遍历。在时间就自然存在消耗。


结论


1.  RelativeLayout会让子View调用2次onMeasure,LinearLayout 在有weight时,也会调用子View2次onMeasure


2. RelativeLayout的子View如果高度和RelativeLayout不同,则会引发效率问题,当子View很复杂时,这个问题会更加严重。如果可以,尽量使用padding代替margin。


3.在不影响层级深度的情况下,使用LinearLayout和FrameLayout而不是RelativeLayout。


最后再思考一下文章开头那个矛盾的问题,为什么Google给开发者默认新建了个RelativeLayout,而自己却在DecorView中用了个LinearLayout。因为DecorView的层级深度是已知而且固定的,上面一个标题栏,下面一个内容栏。采用RelativeLayout并不会降低层级深度,所以此时在根节点上用LinearLayout是效率最高的。而之所以给开发者默认新建了个RelativeLayout是希望开发者能采用尽量少的View层级来表达布局以实现性能最优,因为复杂的View嵌套对性能的影响会更大一些。


4.能用两层LinearLayout,尽量用一个RelativeLayout,在时间上此时RelativeLayout耗时更小。另外LinearLayout慎用layout_weight,也将会增加一倍耗时操作。由于使用LinearLayout的layout_weight,大多数时间是不一样的,这会降低测量的速度。这只是一个如何合理使用Layout的案例,必要的时候,你要小心考虑是否用layout weight。总之减少层级结构,才是王道,让onMeasure做延迟加载,用viewStub,include等一些技巧。


------------ ------ 推荐 ---------- --------


范品社推出了十几款程序员、电影、美剧和物理题材的 极客T恤 单件 ¥59.9、两件减¥12、四件减¥28 ,详见网店商品页介绍。

网店地址: https://fanpinshe.taobao.com/

淘口令: 复制以下红色内容, 然后打开手淘即可购买

范品社,使用¥极客T恤¥抢先预览(长按复制整段文案,打开手机淘宝即可进入活动内容)







请到「今天看啥」查看全文