我的Android进阶之旅------>如何获取Android控件的宽和高

本文转载于:http://blog.csdn.net/johnny901114/article/details/7839512

我们都知道在onCreate()里面获取控件的高度是0,这是为什么呢?我们来看一下示例:

首先我们自己写一个控件,这个控件非常简单:

[java] view plaincopy

  1. public class MyImageView extends ImageView {  
  2.   
  3.     public MyImageView(Context context, AttributeSet attrs) {  
  4.         super(context, attrs);  
  5.     }  
  6.     public MyImageView(Context context) {  
  7.         super(context);  
  8.     }  
  9.       
  10.     @Override  
  11.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
  12.         super.onMeasure(widthMeasureSpec, heightMeasureSpec);  
  13.         System.out.println("onMeasure 我被调用了"+System.currentTimeMillis());  
  14.     }  
  15.       
  16.     @Override  
  17.     protected void onDraw(Canvas canvas) {  
  18.         super.onDraw(canvas);  
  19.         System.out.println("onDraw 我被调用了"+System.currentTimeMillis());  
  20.     }  
  21.   
  22. }  

布局文件:

[java] view plaincopy

  1. <com.test.MyImageView  
  2.     android:id="@+id/imageview"  
  3.     android:layout_width="wrap_content"  
  4.     android:layout_height="wrap_content"  
  5.     android:src="@drawable/test" />  

测试的Activity的onCreate():

[java] view plaincopy

  1. @Override  
  2. public void onCreate(Bundle savedInstanceState) {  
  3.     super.onCreate(savedInstanceState);  
  4.     setContentView(R.layout.main);          
  5.     System.out.println("执行完毕.."+System.currentTimeMillis());  
  6. }  

现在我们现在来看一下结果:

说明等onCreate方法执行完了,我们定义的控件才会被度量(measure),所以我们在onCreate方法里面通过view.getHeight()获取控件的高度或者宽度肯定是0,因为它自己还没有被度量,也就是说他自己都不知道自己有多高,而你这时候去获取它的尺寸,肯定是不行的.

现在碰到这个问题我们不能不解决,在网上找到了如下办法:

[java] view plaincopy

  1. //------------------------------------------------方法一  
  2. int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);  
  3. int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);  
  4. imageView.measure(w, h);  
  5. int height =imageView.getMeasuredHeight();  
  6. int width =imageView.getMeasuredWidth();  
  7. textView.append("\n"+height+","+width);  
  8.   
  9.   
  10.   
  11.   
  12. //-----------------------------------------------方法二  
  13. ViewTreeObserver vto = imageView.getViewTreeObserver();  
  14. vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {  
  15.     public boolean onPreDraw() {  
  16.         int height = imageView.getMeasuredHeight();  
  17.         int width = imageView.getMeasuredWidth();  
  18.         textView.append("\n"+height+","+width);  
  19.         return true;  
  20.     }  
  21. });  
  22. //-----------------------------------------------方法三     
  23. ViewTreeObserver vto2 = imageView.getViewTreeObserver();    
  24. vto2.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {  
  25.     @Override    
  26.     public void onGlobalLayout() {  
  27.         imageView.getViewTreeObserver().removeGlobalOnLayoutListener(this);    
  28.         textView.append("\n\n"+imageView.getHeight()+","+imageView.getWidth());  
  29.     }    
  30. });    

这三个方法是哪里找到现在已经忘了.

现在要讨论的是当我们需要时候使用哪个方法呢?

现在把测试的Activity改成如下:

[java] view plaincopy

  1. @Override  
  2.   public void onCreate(Bundle savedInstanceState) {  
  3.       super.onCreate(savedInstanceState);  
  4.       setContentView(R.layout.main);  
  5.       final ImageView imageView = (ImageView) findViewById(R.id.imageview);        
  6.         
  7.       //------------------------------------------------方法一  
  8.       int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);  
  9.       int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);  
  10.       imageView.measure(w, h);  
  11.       int height =imageView.getMeasuredHeight();  
  12.       int width =imageView.getMeasuredWidth();  
  13.       textView.append("\n"+height+","+width);  
  14.         
  15.       System.out.println("执行完毕.."+System.currentTimeMillis());  
  16.   }  

接着来看下面几种方式输出结果:

把测试Activity改成如下:

[java] view plaincopy

  1. @Override  
  2. public void onCreate(Bundle savedInstanceState) {  
  3.     super.onCreate(savedInstanceState);  
  4.     setContentView(R.layout.main);  
  5.     final ImageView imageView = (ImageView) findViewById(R.id.imageview);  
  6. -----------------------------------------------方法二  
  7.     ViewTreeObserver vto = imageView.getViewTreeObserver();  
  8.     vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {  
  9.         public boolean onPreDraw() {  
  10.             int height = imageView.getMeasuredHeight();  
  11.             int width = imageView.getMeasuredWidth();  
  12.             textView.append("\n"+height+","+width);  
  13.             return true;  
  14.         }  
  15.     });  
  16. }  

结果如下:

方法三就不再测试了同方法二!!!

那么方法而和方法三在执行上有什么区别呢?

我们在布局文件中加入一个TextView来记录这个控件的宽高.

[java] view plaincopy

  1. <ScrollView  
  2.     android:layout_width="wrap_content"  
  3.     android:layout_height="wrap_content" >  
  4.   
  5.     <TextView  
  6.         android:id="@+id/text"  
  7.         android:layout_width="wrap_content"  
  8.         android:layout_height="wrap_content" />  
  9. </ScrollView>  

先来测试方法而:

[java] view plaincopy

  1. @Override  
  2. public void onCreate(Bundle savedInstanceState) {  
  3.     super.onCreate(savedInstanceState);  
  4.     setContentView(R.layout.main);  
  5.     final ImageView imageView = (ImageView) findViewById(R.id.imageview);  
  6. -----------------------------------------------方法二  
  7.     ViewTreeObserver vto = imageView.getViewTreeObserver();  
  8.     vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {  
  9.         public boolean onPreDraw() {  
  10.             int height = imageView.getMeasuredHeight();  
  11.             int width = imageView.getMeasuredWidth();  
  12.             textView.append("\n"+height+","+width);  
  13.             return true;  
  14.         }  
  15.     });  
  16. }  

结果如下:

我们再来测试方法三

[java] view plaincopy

  1. @Override  
  2. public void onCreate(Bundle savedInstanceState) {  
  3.     super.onCreate(savedInstanceState);  
  4.     setContentView(R.layout.main);  
  5.     final ImageView imageView = (ImageView) findViewById(R.id.imageview);  
  6.     //-----------------------------------------------方法三     
  7.     ViewTreeObserver vto2 = imageView.getViewTreeObserver();    
  8.     vto2.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {  
  9.         @Override    
  10.         public void onGlobalLayout() {  
  11.             imageView.getViewTreeObserver().removeGlobalOnLayoutListener(this);    
  12.             textView.append("\n\n"+imageView.getHeight()+","+imageView.getWidth());  
  13.         }    
  14.     });    
  15. }  

输出结果如下:

我想这方法二和方法三之间的区别就不用说了吧.

  总结:那么需要获取控件的宽高该用那个方法呢?

方法一: 比其他的两个方法多了一次计算,也就是多调用了一次onMeasure()方法,该方法虽然看上去简单,但是如果要目标控件计算耗时比较大的话(如listView等),不建议使用.

方法二,它的回调方法会调用很多次,并且滑动TextView的时候任然会调用,所以不建议使用.

方法三,比较合适.

当然,实际应用的时候需要根据实际情况而定.

时间: 2024-11-01 21:12:52

我的Android进阶之旅------&gt;如何获取Android控件的宽和高的相关文章

我的Android进阶之旅------&amp;gt;如何解决Android 5.0中出现的警告: Service Intent must be explicit:

我的Android进阶之旅-->如何解决Android 5.0中出现的警告: java.lang.IllegalArgumentException: Service Intent must be explicit: Intent { act=com.xtc.kuwo.watch.MUSIC_PLAY_SERVICE (has extras) } 1.错误描述 今天在Android4.4 的小米4手机上运行我的程序的时候没有报错,而在Android 5.1的华为P7上运行我的程序的时候报了以下的错

我的Android进阶之旅------&amp;gt;如何获取系统中定义了那些权限

在Window控制台中输入如下命令可以看到Android系统中列出的所有权限(如果自定义权限注册成功,在这里也会找到这些自定义的权限) adb shell pm list permissions C:\Users\Administrator>adb shell pm list permissions All Permissions: permission:android.permission.INTERNAL_SYSTEM_WINDOW permission:android.permission

【我的Android进阶之旅】如何隐藏Android中EditText控件的默认下划线

Android EditText控件是经常使用的控件,但是有时候我们并不需要它的一些默认的属性,比如说下划线,因为有时候这样的默认下划线看起来特别怪异,和其他控件在一起搭配的时候不协调,因此有时候就需要去掉默认的下划线. 下面先看看默认的效果. 代码如下 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.

我的Android进阶之旅------&amp;gt;关于使用Android Studio替换App的launcher图标之后仍然显示默认的ic_launcher图标的解决方法

前言 最近做了一个App,之前开发该App的时候一直以来都是默认的launcher图标启动的, 今天美工换了一个App的launcher 图标,因此在Android Studio中将默认的lancher图标替换成美工的图标,运行到真机后,发现一直没有替换成功. 替换 下面是替换lancher图标的代码,将如下默认的代码 <application android:name=".app.QQNewsApp" android:allowBackup="true" a

我的Android进阶之旅------&amp;gt;如何将Android源码导入Eclipse中来查看(非常实用)

Android源码下载完成的目录结构如如所示: step1:将.classpath文件拷贝到源代码的根目录 Android源码支持多种IDE,如果是针对APP层做开发的话,建议大家使用Eclipse开发环境.在源码路径(~/development/ide/eclipse)下,Android提供了.classpath配置文件(备注:该文件为隐藏文件). cp development/ide/eclipse/.classpath ./ chmod u+w .classpath # Make the

我的Android进阶之旅------&amp;gt;ListView中android:cacheColorHint,android:listSelector属性作用 .

            ( 本文转载于:http://blog.csdn.net/stonecao/article/details/6216449)        自定义listview的时候,当你不使用android:cacheColorHint="#00000000"会出现下面选中一个空间黑色底色的情况,破坏整体美观度:                当你不使用android:listSelector属性,默认会显示选中的item为橙黄底色,有时候我们需要去掉这种效果:      

我的Android进阶之旅------&amp;gt;解决Jackson等第三方转换Json的开发包在开启混淆后转换的实体类数据都是null的bug

1.错误描述 今天测试人员提了一个bug,说使用我们的app出现了闪退的bug,后来通过debug断点调试,发现我们的app转换服务器发送过来的json数据后,都是为null.而之前已经提测快一个月的功能,一直都是稳定的,为什么现在会报java.lang.NullPointerException. 2.错误原因 原来我提测了一个月的APP版本一直没有打开混淆开关,而出问题的这个APP版本在即将要发布出去的时候打开了混淆开关.这样的话,我那些要通过转换json数据为bean实体类,因为没有在pro

我的Android进阶之旅------&amp;gt;Android疯狂连连看游戏的实现之状态数据模型(三)

对于游戏玩家而言,游戏界面上看到的"元素"千变万化:但是对于游戏开发者而言,游戏界面上的元素在底层都是一些数据,不同数据所绘制的图片有所差异而已.因此建立游戏的状态数据模型是实现游戏逻辑的重要步骤. 1.定义数据模型 连连看的界面是一个NxM的"网格",每个网格上显示一张图片.而这个网格只需要一个二维数组来定义即可,而每个网格上所显示的图片,对于底层数据模型来说,不同的图片对于着不同的数值即可. 对于上图所示的数据模型,只要让数值为0的网格上不绘制图片,其他数值的网

我的Android进阶之旅------&amp;gt;介绍一款集录制与剪辑为一体的屏幕GIF 动画制作工具 GifCam

由于上一篇文章:我的Android进阶之旅------>Android之动画之Frame Animation实例 中展示的是Frame动画效果,但是之前我是将图片截取下来,不好说明确切的动画过程,因此今天百度了一下gif动画的制作工具,找到了这款不错的软件GifCam.然后我使用该软件将动画制作成gif动画,然后更新了上一篇文章上传了新制作的gif动画,使博客的说明效果更佳. 该软件的界面如下所示: 该软件的操作步骤如下: 1.拖动 GifCam 标题栏,改变其位置,并调整 GifCam 窗口大