zoukankan      html  css  js  c++  java
  • MeasureSpec介绍及使用详解

      一个MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureSpec代表了一组宽度和高度的要求。一个MeasureSpec由大小和模式组成。它有三种模式:UNSPECIFIED(未指定),父元素部队自元素施加任何束缚,子元素可以得到任意想要的大小;EXACTLY(完全),父元素决定自元素的确切大小,子元素将被限定在给定的边界里而忽略它本身大小;AT_MOST(至多),子元素至多达到指定大小的值。

     

      它常用的三个函数:

      1.static int getMode(int measureSpec):根据提供的测量值(格式)提取模式(上述三个模式之一)

      2.static int getSize(int measureSpec):根据提供的测量值(格式)提取大小值(这个大小也就是我们通常所说的大小)

      3.static int makeMeasureSpec(int size,int mode):根据提供的大小值和模式创建一个测量值(格式)


      这个类的使用呢,通常在view组件的onMeasure方法里面调用但也有少数例外,看看几个例子:

     

      a.首先一个我们常用到的一个有用的函数,View.resolveSize(int size,int measureSpec)

     

    8421     public static int resolveSize(int size, int measureSpec) {
    8422 int result = size;
    8423 int specMode = MeasureSpec.getMode(measureSpec);
    8424 int specSize = MeasureSpec.getSize(measureSpec);
    8425 switch (specMode) {
    8426 case MeasureSpec.UNSPECIFIED:
    8427 result = size;
    8428 break;
    8429 case MeasureSpec.AT_MOST:
    8430 result = Math.min(size, specSize);
    8431 break;
    8432 case MeasureSpec.EXACTLY:
    8433 result = specSize;
    8434 break;
    8435 }
    8436 return result;
    8437 }

      上面既然要用到measureSpec值,那自然表示这个函数通常是在onMeasure方法里面调用的。简单说一下,这个方法的主要作用就是根据你提供的大小和模式,返回你想要的大小值,这个里面根据传入模式的不同来做相应的处理。

      再看看MeasureSpec.makeMeasureSpec方法,实际上这个方法很简单:

    9023         public static int makeMeasureSpec(int size, int mode) {
    9024 return size + mode;
    9025 }

      这样大家不难理解size跟measureSpec区别了。看看它的使用吧,ListView.measureItem(View child)

     

    2464     private void measureItem(View child) {
    2465 ViewGroup.LayoutParams p = child.getLayoutParams();
    2466 if (p == null) {
    2467 p = new ViewGroup.LayoutParams(
    2468 ViewGroup.LayoutParams.MATCH_PARENT,
    2469 ViewGroup.LayoutParams.WRAP_CONTENT);
    2470 }
    2471
    2472 int childWidthSpec = ViewGroup.getChildMeasureSpec(mWidthMeasureSpec,
    2473 mListPadding.left + mListPadding.right, p.width);
    2474 int lpHeight = p.height;
    2475 int childHeightSpec;
    2476 if (lpHeight > 0) {
    2477 childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight, MeasureSpec.EXACTLY);
    2478 } else {
    2479 childHeightSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
    2480 }
    2481 child.measure(childWidthSpec, childHeightSpec);
    2482 }

      measureSpec方法通常在ViewGroup中用到,它可以根据模式(MeasureSpec里面的三个)可以调节子元素的大小。

      注意,使用EXACTLY和AT_MOST通常是一样的效果,如果你要区别他们,那么你就要使用上面的函数View.resolveSize(int size,int measureSpec)返回一个size值,然后使用你的view调用setMeasuredDimension(int,int)函数。

     

    8406     protected final void setMeasuredDimension(int measuredWidth, int measuredHeight) {
    8407 mMeasuredWidth = measuredWidth;
    8408 mMeasuredHeight = measuredHeight;
    8409
    8410 mPrivateFlags |= MEASURED_DIMENSION_SET;
    8411 }

      然后你调用view.getMeasuredWidth,view.getMeasuredHeigth 返回的就是上面函数里的mMeasuredWidth,mMeasuredHeight的值。



      

     

  • 相关阅读:
    互动教程,让你5分钟掌握 Flexbox 布局模式
    Fixed Responsive Nav – 响应式的单页网站导航插件
    创意无限!一组网页边栏过渡动画【附源码下载】
    12款界面精美的 HTML5 & CSS3 网站模板
    Twproject Gantt – 开源的 JavaScript 甘特图组件
    真是好东西!一组动感的页面加载动画效果
    Method Draw – 很好用的 SVG 在线编辑器
    CSS Vocabulary – CSS 词汇表,你都掌握了吗?
    前端精选文摘:BFC 神奇背后的原理
    Vis.js – 基于浏览器的动态 JavaScript 可视化库
  • 原文地址:https://www.cnblogs.com/slider/p/2266538.html
Copyright © 2011-2022 走看看