ViewGroup.LayoutParams 和 MeasureSpec

1.LayoutParams

 LayoutParams  是ViewGroup的内部静态类 ,ViewGroup的子类(如RelativeLayout,LinearLayout,FrameLayout)都有其对应的     ViewGroup.LayoutParams的子类,如RelativeLayoutParams

 LayoutParams 的作用:指定视图View 的高度(height) 和 宽度(width)等布局参数,具体如下:

参数解释
具体值dp / px
fill_parent强制性使子视图的大小扩展至与父视图大小相等(不含 padding )
match_parent与fill_parent相同,用于Android 2.3 & 之后版本
wrap_content自适应大小,强制性地使视图扩展以便显示其全部内容(含 padding )

对应于xml如下:

android:layout_height="wrap_content"   //自适应大小  
android:layout_height="match_parent"   //与父视图等高  
android:layout_height="fill_parent"    //与父视图等高  
android:layout_height="20dp"         //精确设置高度值为 20dp

下面是一个LinearLayoutParams的使用示例:

布局文件如下,只有一个TextView

<LinearLayout
    android:id="@+id/linear"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    xmlns:android="http://schemas.android.com/apk/res/android">
    <TextView
        android:layout_width="200dp"
        android:layout_height="30dp"
        android:text="第一个TextView"
        android:gravity="center"
        android:background="#6f00"
        />
</LinearLayout>

Activity代码如下:

public class MainActivity extends AppCompatActivity {

    private LinearLayout mRootView;
    private LinearLayout mLinearLayout;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mRootView = (LinearLayout) findViewById(R.id.linear);

        // 把 LinearLayout 添加到布局里面
        mLinearLayout = new LinearLayout(MainActivity.this);
        mLinearLayout.setBackgroundColor(Color.parseColor("#00ff00"));

        LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.WRAP_CONTENT);
        mLinearLayout.setLayoutParams(layoutParams);
        mRootView.addView(mLinearLayout);

        // 第二步,把TextView 添加到 LinearLayout,以增加一个TextView
        TextView textView = new TextView(MainActivity.this);
        textView.setText("第二个TextView");
        textView.setTextSize(20);
        textView.setBackgroundColor(Color.parseColor("#ff0000"));
        textView.setGravity(Gravity.CENTER);
        mLinearLayout.addView(textView);

        // 设置TextView大小为具体值
        LinearLayout.LayoutParams textParams = new LinearLayout.LayoutParams(textView.getLayoutParams());
        textParams.width = dip2px(MainActivity.this,200);
        textParams.height = dip2px(MainActivity.this,30);
        textView.setLayoutParams(textParams);

    }

    private int dip2px(Context context, float dipValue)
    {
        Resources r = context.getResources();
        return (int) TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_DIP, dipValue, r.getDisplayMetrics());
    }


}

2.MeasureSpec

测量规格(MeasureSpec) = 测量模式(mode) + 测量大小(size)

其中,测量模式(Mode)的类型有3种:UNSPECIFIED、EXACTLY 和 AT_MOST。具体如下:

 

MeasureSpec 被封装在View类中的一个内部类里:MeasureSpec

MeasureSpec类 用1个变量封装了2个数据(size,mode)通过使用二进制,将测量模式(mode) & 测量大小(size)打包成一个int值来,并提供了如下的打包/解包的方法

/** * MeasureSpec类的具体使用 **/ //

   获取测量模式(Mode)                                int specMode = MeasureSpec.getMode(measureSpec)

   获取测量大小(Size)                                  int specSize = MeasureSpec.getSize(measureSpec)

   通过Mode 和 Size 生成新的SpecMode         int measureSpec=MeasureSpec.makeMeasureSpec(size, mode);

 

MeasureSpec值的计算:子View的MeasureSpec值根据子View的布局参数(LayoutParams)和父容器的MeasureSpec值计算得来的,具体计算逻辑封装在getChildMeasureSpec(),即view的大小由父viewMeasureSpec值 和 子viewLayoutParams属性 共同决定。

/**

  * 源码分析:getChildMeasureSpec()

  * 作用:根据父视图的MeasureSpec & 布局参数LayoutParams,计算单个子ViewMeasureSpec

  * 注:子view的大小由父viewMeasureSpec viewLayoutParams属性 共同决定

  **/

 

    public static int getChildMeasureSpec(int spec, int padding, int childDimension) { 

         //参数说明

         * @param spec view的详细测量值(MeasureSpec)

         * @param padding view当前尺寸的的内边距和外边距(padding,margin)

         * @param childDimension 子视图的布局参数(宽/高)

            //view的测量模式

            int specMode = MeasureSpec.getMode(spec);    

            //view的大小

            int specSize = MeasureSpec.getSize(spec);    

            //通过父view计算出的子view = 父大小-边距(父要求的大小,但子view不一定用这个值)  

            int size = Math.max(0, specSize - padding); 

            //view想要的实际大小和模式(需要计算) 

            int resultSize = 0

            int resultMode = 0;  

            //通过父viewMeasureSpec和子viewLayoutParams确定子view的大小 

            // 当父view的模式为EXACITY时,父view强加给子view确切的值

           //一般是父view设置为match_parent或者固定值的ViewGroup

            switch (specMode) { 

            case MeasureSpec.EXACTLY: 

                // 当子viewLayoutParams>0,即有确切的值 

                if (childDimension >= 0) { 

                    //view大小为子自身所赋的值,模式大小为EXACTLY 

                    resultSize = childDimension; 

                    resultMode = MeasureSpec.EXACTLY; 

                // 当子viewLayoutParamsMATCH_PARENT(-1) 

                } else if (childDimension == LayoutParams.MATCH_PARENT) { 

                    //view大小为父view大小,模式为EXACTLY 

                    resultSize = size; 

                    resultMode = MeasureSpec.EXACTLY; 

                // 当子viewLayoutParamsWRAP_CONTENT(-2)     

                } else if (childDimension == LayoutParams.WRAP_CONTENT) { 

                    //view决定自己的大小,但最大不能超过父view,模式为AT_MOST 

                    resultSize = size; 

                    resultMode = MeasureSpec.AT_MOST; 

                } 

                break

            // 当父view的模式为AT_MOST时,父view强加给子view一个最大的值。(一般是父view设置为wrap_content 

            case MeasureSpec.AT_MOST: 

                // 道理同上 

                if (childDimension >= 0) { 

                    resultSize = childDimension; 

                    resultMode = MeasureSpec.EXACTLY; 

                } else if (childDimension == LayoutParams.MATCH_PARENT) { 

                    resultSize = size; 

                    resultMode = MeasureSpec.AT_MOST; 

                } else if (childDimension == LayoutParams.WRAP_CONTENT) { 

                    resultSize = size; 

                    resultMode = MeasureSpec.AT_MOST; 

                } 

                break

            // 当父view的模式为UNSPECIFIED时,父容器不对view有任何限制,要多大给多大

            // 多见于ListViewGridView 

            case MeasureSpec.UNSPECIFIED: 

                if (childDimension >= 0) { 

                    // view大小为子自身所赋的值 

                    resultSize = childDimension; 

                    resultMode = MeasureSpec.EXACTLY; 

                } else if (childDimension == LayoutParams.MATCH_PARENT) { 

                    // 因为父viewUNSPECIFIED,所以MATCH_PARENT的话子类大小为

                    resultSize = 0

                    resultMode = MeasureSpec.UNSPECIFIED; 

                } else if (childDimension == LayoutParams.WRAP_CONTENT) { 

                    // 因为父viewUNSPECIFIED,所以WRAP_CONTENT的话子类大小为0  

                    resultSize = 0

                    resultMode = MeasureSpec.UNSPECIFIED; 

                } 

                break

            } 

            return MeasureSpec.makeMeasureSpec(resultSize, resultMode); 

        }

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,以下是一个可以测量未绘制过的View宽高的Java工具类: ```java import android.view.View; import android.view.View.MeasureSpec; public class ViewUtils { /** * 测量未绘制过的View的宽高 * * @param view 待测量的View * @return int[宽, 高] */ public static int[] measureView(View view) { int[] result = new int[2]; // 获取LayoutParams ViewGroup.LayoutParams lp = view.getLayoutParams(); if (lp == null) { lp = new ViewGroup.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); } // 生成MeasureSpecMeasureSpec的值由specMode和specSize组合而成 int widthSpec = ViewGroup.getChildMeasureSpec(0, 0, lp.width); int heightSpec; if (lp.height > 0) { heightSpec = MeasureSpec.makeMeasureSpec(lp.height, MeasureSpec.EXACTLY); } else { heightSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED); } // 执行measure方法,测量View的宽高 view.measure(widthSpec, heightSpec); result[0] = view.getMeasuredWidth(); result[1] = view.getMeasuredHeight(); return result; } } ``` 这个工具类中的`measureView()`方法接收一个View对象作为参数,返回该View的宽高,如果该View没有设置LayoutParams,则默认使用MATCH_PARENT和WRAP_CONTENT。MeasureSpec是Android中用于测量View的尺寸的类,specMode决定了View的大小是精确值、最大值或者是未指定;specSize则是View的大小。在代码中,我们通过getChildMeasureSpec()方法获取宽度MeasureSpec,并根据LayoutParams中的height设置高度MeasureSpec。最后通过`view.measure()`方法测量View的宽高,并将结果存入数组中返回。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值