android如何获取view在布局中的高度与宽度详解

所属分类: 软件编程 / Android 阅读数: 41
收藏 0 赞 0 分享

前言

可能很多情况下,我们都会有在activity中获取view 的尺寸大小(宽度和高度)的需求。面对这种情况,很多同学立马反应:这么简单的问题,还用你说?你是不是傻。。然后立马写下getWidth()、getHeight()等方法,洋洋得意的就走了。然而事实就是这样的吗?实践证明,我们这样是获取不到View的宽度和高度大小的。

当我们在 onCreate() 方法中获取某个 View 组件的宽度和高度,直接调用 getWidth()、getHeight()、getMeasuredWidth()、getMeasuredHeight() 方法只会得到 0。这是什么原因呢?下面来一起看看吧

实现方法

一、使用 View.measure 测量 View

该方法测量的宽度和高度可能与视图绘制完成后的真实的宽度和高度不一致。

int width = View.MeasureSpec.makeMeasureSpec(0,
  View.MeasureSpec.UNSPECIFIED);
int height = View.MeasureSpec.makeMeasureSpec(0,
  View.MeasureSpec.UNSPECIFIED);
view.measure(width, height);
view.getMeasuredWidth(); // 获取宽度
view.getMeasuredHeight(); // 获取高度

二、使用 ViewTreeObserver. OnPreDrawListener 监听事件

在视图将要绘制时调用该监听事件,会被调用多次,因此获取到视图的宽度和高度后要移除该监听事件。

view.getViewTreeObserver().addOnPreDrawListener(
  new ViewTreeObserver.OnPreDrawListener() {
 
 @Override
 public boolean onPreDraw() {
  view.getViewTreeObserver().removeOnPreDrawListener(this);
  view.getWidth(); // 获取宽度
  view.getHeight(); // 获取高度
  return true;
 }
});

三、使用 ViewTreeObserver. OnGlobalLayoutListener 监听事件

在布局发生改变或者某个视图的可视状态发生改变时调用该事件,会被多次调用,因此需要在获取到视图的宽度和高度后执行 remove 方法移除该监听事件。

view.getViewTreeObserver().addOnGlobalLayoutListener(
  new ViewTreeObserver.OnGlobalLayoutListener() {
 
 @Override
 public void onGlobalLayout() {
  if (Build.VERSION.SDK_INT >= 16) {
   view.getViewTreeObserver()
     .removeOnGlobalLayoutListener(this);
  }
  else {
   view.getViewTreeObserver()
     .removeGlobalOnLayoutListener(this);
  }
  view.getWidth(); // 获取宽度
  view.getHeight(); // 获取高度
 }
});

四、重写 View 的 onSizeChanged 方法

在视图的大小发生改变时调用该方法,会被多次调用,因此获取到宽度和高度后需要考虑禁用掉代码。
该实现方法需要继承 View,且多次被调用,不建议使用。

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
 super.onSizeChanged(w, h, oldw, oldh);
 
 view.getWidth(); // 获取宽度
 view.getHeight(); // 获取高度
}

五、重写 View 的 onLayout 方法

该方法会被多次调用,获取到宽度和高度后需要考虑禁用掉代码。

该实现方法需要继承 View,且多次被调用,不建议使用。

@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
 super.onLayout(changed, l, t, r, b);
 
 view.getWidth(); // 获取宽度
 view.getHeight(); // 获取高度
}

六、使用 View.OnLayoutChangeListener 监听事件(API >= 11)

在视图的 layout 改变时调用该事件,会被多次调用,因此需要在获取到视图的宽度和高度后执行 remove 方法移除该监听事件。

view.addOnLayoutChangeListener(
  new View.OnLayoutChangeListener() {
 
 @Override
 public void onLayoutChange(View v, int l, int t, int r, int b,
   int oldL, int oldT, int oldR, int oldB) {
  view.removeOnLayoutChangeListener(this);
  view.getWidth(); // 获取宽度
  view.getHeight(); // 获取高度
  }
});

七、使用 View.post() 方法

Runnable 对象中的方法会在 View 的 measure、layout 等事件完成后触发。

UI 事件队列会按顺序处理事件,在 setContentView() 被调用后,事件队列中会包含一个要求重新 layout 的 message,所以任何 post 到队列中的 Runnable 对象都会在 Layout 发生变化后执行。

该方法只会执行一次,且逻辑简单,建议使用。

view.post(new Runnable() {
 
 @Override
 public void run() {
  view.getWidth(); // 获取宽度
  view.getHeight(); // 获取高度
 }
});

以上为转载内容,个人学习收藏记录

下面是自己的学习记录。

首先第一个方法,以前用过,确实不准确,猜测是应该是因为参数没有用好,因为参数只使用UNSPECIFIED未指定的测量方式,一般像Wrap_Content,才是该测量方式。

这里贴一个比较好用的, AndroidUtilCode收藏的方法。

 public static int[] measureView(final View view) {
  ViewGroup.LayoutParams lp = view.getLayoutParams();
  if (lp == null) {
   lp = new ViewGroup.LayoutParams(
     ViewGroup.LayoutParams.MATCH_PARENT,
     ViewGroup.LayoutParams.WRAP_CONTENT
   );
  }
  int widthSpec = ViewGroup.getChildMeasureSpec(0, 0, lp.width);
  int lpHeight = lp.height;
  int heightSpec;
  if (lpHeight > 0) {
   heightSpec = View.MeasureSpec.makeMeasureSpec(lpHeight, View.MeasureSpec.EXACTLY);
  } else {
   heightSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
  }
  view.measure(widthSpec, heightSpec);
  return new int[]{view.getMeasuredWidth(), view.getMeasuredHeight()};
 }

然后是自己在做自定义view的时候,需要在一次add代码创建的view,使用上面的方法无法获得宽高,因为我使用的是ScrollView。像在自定义中,加载一次布局,应该选中最后一个post的方法最为使用。

另外还用的多的,应该是第三种方式,一般在外部使用,比如需要等待Recyclerview绘制完成后进行的操作。

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

更多精彩内容其他人还在看

使用ViewPager实现android软件使用向导功能实现步骤

现在的大部分android软件,都是使用说明,就是第一次使用该软件时,会出现向导,可以左右滑动,然后就进入应用的主界面了,下面我们就实现这个功能
收藏 0 赞 0 分享

android在异步任务中关闭Cursor的代码方法

android在异步任务中如何关闭Cursor?在我们开发应用的时候,很多时候会遇到这种问题,下面我们就看看代码如何实现
收藏 0 赞 0 分享

Android自定义桌面功能代码实现

android自定义桌面其实很简单,看一个例子就明白了
收藏 0 赞 0 分享

android将图片转换存到数据库再从数据库读取转换成图片实现代码

有时候我们想把图片存入到数据库中,尽管这不是一种明智的选择,但有时候还是不得以会用到,下面说说将图片转换成byte[]数组存入到数据库中去,并从数据库中取出来转换成图像显示出来
收藏 0 赞 0 分享

TextView显示系统时间(时钟功能带秒针变化

用System.currentTimeMillis()可以获取系统当前的时间,我们可以开启一个线程,然后通过handler发消息,来实时的更新TextView上显示的系统时间,可以做一个时钟的功能
收藏 0 赞 0 分享

Android用ListView显示SDCard文件列表的小例子

本文简单实现了用ListView显示SDCard文件列表,目录的回退等功能暂不讨论,获取文件列表,files即为所选择目录下的所有文件列表
收藏 0 赞 0 分享

Android拦截外拨电话程序示例

这篇文章主要介绍了Android拦截外拨电话的示例,大家参考使用吧
收藏 0 赞 0 分享

通过Html网页调用本地安卓(android)app程序代码

如何使用html网页和本地app进行传递数据呢?经过研究,发现还是有方法的,总结了一下,大致有一下几种方式
收藏 0 赞 0 分享

android Textview文字监控(Textview使用方法)

以手机号充值为例,当用户输入最后一位数时候,进行汇率的变换,本文就实现类似这样的功能
收藏 0 赞 0 分享

Android ListView长按弹出菜单二种实现方式示例

这篇文章主要介绍了Android ListView长按弹出菜单的方法,大家参考实现
收藏 0 赞 0 分享
查看更多