为什么嵌套权重对性能不利?备择方案?


160

我编写了几个布局文件,在其中我使用该layout_weight属性在不同视图之间创建比率。

在某个时候,我开始收到有关嵌套权重的棉绒警告。

因此,我想知道为什么嵌套权重对性能不利,并且是否存在一种更有效的方法来在视图尺寸之间创建恒定比率,该比例可用于不同的屏幕尺寸,而无需指定很多尺寸dpi值在几个布局文件中(对于不同的屏幕尺寸,我是说)。

谢谢!


Answers:


140

嵌套权重不利于性能,因为:

布局权重要求对小部件进行两次测量。当权重为非零的LinearLayout嵌套在权重为非零的另一个LinearLayout中时,度量的数量将呈指数增长。

最好使用RelativeLayout并根据其他视图的位置调整视图,而不使用特定的dpi值。


87
我想这是消息的目的,这是需要注意的好事情。我指出的是,如果涉及的指数很小,则指数影响仍然很小。对于较小的嵌套深度,使用执行此操作所需的CPU,就好像拥有一整周都在宠爱自己的工作马,只在周日带出去散步一样。不过,对于较大深度的嵌套,这是一个很好的观点。
卡尔

14
RelativeLayout还需要进行两次测量以确保其所有子布局正确,因此将具有布局权重的LinearLayout更改为RelativeLayout可能不会提高性能。
Piasy

相对布局并不总是有效。如果您需要构造成比例的小部件
Abdurakhmon,

67

更新:据我们所知,API级别26不推荐使用百分比支持库。这ConstraintLayout是实现相同平面xml结构的新方法。

更新的Github项目

更新的样本:

<android.support.constraint.ConstraintLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/fifty_thirty"
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:background="#ffff8800"
        android:gravity="center"
        android:text="@string/fifty_fifty_text"
        android:textColor="@android:color/white"
        app:layout_constraintHeight_default="percent"
        app:layout_constraintHeight_percent="0.5"
        android:textSize="25sp"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintWidth_default="percent"
        app:layout_constraintWidth_percent="0.5" />

    <TextView
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:background="#ffff5566"
        android:gravity="center"
        android:text="@string/fifty_fifty_text"
        android:textColor="@android:color/white"
        android:textSize="25sp"
        app:layout_constraintHeight_default="percent"
        app:layout_constraintHeight_percent="0.5"
        app:layout_constraintLeft_toRightOf="@id/fifty_thirty"
        app:layout_constraintTop_toBottomOf="@id/fifty_thirty"
        app:layout_constraintWidth_default="percent"
        app:layout_constraintWidth_percent="0.5" />

</android.support.constraint.ConstraintLayout>

更新:好消息android百分比支持库解决了我们的性能问题和嵌套凌乱的加权LinearLayout

compile 'com.android.support:percent:23.0.0'

演示在这里

考虑这个简单的布局来演示相同的内容。

支持libray演示的百分比

<android.support.percent.PercentRelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView
        android:id="@+id/fifty_huntv"
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:background="#ff7acfff"
        android:text="20% - 50%"
        android:textColor="@android:color/white"
        app:layout_heightPercent="20%"
        app:layout_widthPercent="50%" />
    <TextView
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:layout_toRightOf="@id/fifty_huntv"
        android:background="#ffff5566"
        android:text="80%-50%"
        app:layout_heightPercent="80%"
        app:layout_widthPercent="50%"
        />

</android.support.percent.PercentRelativeLayout>

避免将性能降低器LinearLayout与权重嵌套在一起。


@dan是的,考虑到我们嵌套了带有权重的线性布局。
nitesh

3
“此类已在API级别26.0.0-beta1中弃用。请考虑改用ConstraintLayout和关联的布局。” developer.android.com/reference/android/support/percent/...
saiyancoder

7
我不喜欢ConstraintLayout。它对我而言并不直观
卡森·霍尔兹海默

8
ConstraintLayout对我来说是如此困难
BertKing

1
也许苹果给出的关于自动布局约束的解释更加清晰,并且由于逻辑相同,因此可以提供帮助。不幸的是,我发现droid的ConstraintLayout使用起来比iOS的AutoLayout更沉重/冗长
-AdricoM

46

我认为(对此我可能会感到恼火),但我仍然认为我的手机具有四核处理器,可以与大多数人的家用PC媲美(如果不能完全销毁)。

我也认为这种硬件功能是手机的未来。

因此,我得出一个结论,只要您不被嵌套困扰(在MHO中,布局的深度绝不能超过4层,如果是错的话),您的手机可能就不会在乎关于减肥。

您可以做很多事情,这些都会对性能产生更深远的影响,然后让您的处理器担心一些额外的数学运算。

(请注意,我有点幽默,所以不要从这篇文章中过分认真对待,否则会认为您首先需要优化其他东西,而担心2-3级深的重量并没有帮助你的健康)


2
并基本同意,但听说过iphone的平均使用量(包括支持其使用的Web服务/站点)与美国普通家用冰箱每年的能量消耗大致相同。因此,作为开发人员,我们有责任考虑这种环境影响。显然,这始终是一种平衡的行为:时间,成本,性能,稳定性,总的来说,我同意您的观点-但我认为我们也应该考虑这种影响。显然,维护/可扩展性也在这里出现。无论如何-提出要点,谢谢。
MemeDeveloper

意识到有问题的特定点是关于在设备而不是Web上进行处理,但是我的评论是关于作为开发人员而不是OP的更多细节的优先事项的一般性观点。
MemeDeveloper

11

嵌套权重不好的主要原因是,当布局中的子项具有权重时,必须对其进行两次测量(我认为这是皮棉警告中提到的)。这意味着还包含加权布局的加权布局必须测量四次,并且您添加的每个权重“层”以2的幂来增加度量。

在ICS(API级别14)GridLayout中添加了,它为以前需要权重的许多布局提供了简单且“平坦”的解决方案。如果您是针对较早版本的Android开发的,则移除权重会有些困难,但是RelativeLayout在驾驶室中使用和尽可能扁平化布局通常会消除很多嵌套的权重。


9
我认为使用GridLayout 无法达到相同的结果RelativeLayout。例如GridLayout:“ GridLayout不支持按重量定义的重量原理。通常,因此,不可能将GridLayout配置为在多个组件之间分配多余的空间。”
Timmmm 2012年

从API 21开始,权重的概念已添加到GridLayout中。要支持较旧的android设备,您可以使用v7支持库中的GridLayout。android.support.v7.widget.GridLayout
福音传道者2015年

2

有一个简单的解决方案可以避免使用权重嵌套的LinearLayouts-只需将TableLayoutout与weightSum一起使用,以及将LinearLineLayout与weightSum结合使用-Tablelayout与LinearLayout具有相同的属性(orientation,weightSum,layout_weight等),并且不显示消息-“嵌套权重对性能不利”

例:

 <TableLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical"
        android:weightSum="1">

        <ImageView
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="0.8"/>


        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="0.2"
            android:orientation="horizontal"
            android:weightSum="1">


            <ImageView
                android:layout_height="match_parent"
                android:layout_width="0dp"
                android:layout_weight="0.4"/>

            <TextView
                android:layout_height="match_parent"
                android:layout_width="0dp"
                android:layout_weight="0.6"/>


            </LinearLayout>

    </TableLayout>

1

我认为,唯一的选择是制作一个函数,该函数将被称为onResume并将设置所有大小和位置。无论如何,按重量计,您只能设置大小,而不能设置填充(因此布局变得更加复杂),不能设置textSize(不可能以某种方式进行补偿),更不用说行数了。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.