android view getWidth 和 getHeight 的值为0

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

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

public class MyImageView extends ImageView { 

public MyImageView(Context context, AttributeSet attrs) { 
super(context, attrs); 
} 
public MyImageView(Context context) { 
super(context); 
} 

@Override 
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
super.onMeasure(widthMeasureSpec, heightMeasureSpec); 
System.out.println("onMeasure 我被调用了"+System.currentTimeMillis()); 
} 

@Override 
protected void onDraw(Canvas canvas) { 
super.onDraw(canvas); 
System.out.println("onDraw 我被调用了"+System.currentTimeMillis()); 
} 
}

布局

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

oncreate:

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

结果



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

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

//------------------------------------------------方法一 
int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED); 
int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED); 
imageView.measure(w, h); 
int height =imageView.getMeasuredHeight(); 
int width =imageView.getMeasuredWidth(); 
textView.append("\n"+height+","+width); 




//-----------------------------------------------方法二 
ViewTreeObserver vto = imageView.getViewTreeObserver(); 
vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() { 
public boolean onPreDraw() { 
int height = imageView.getMeasuredHeight(); 
int width = imageView.getMeasuredWidth(); 
textView.append("\n"+height+","+width); 
return true; 
} 
}); 
//-----------------------------------------------方法三 
ViewTreeObserver vto2 = imageView.getViewTreeObserver(); 
vto2.addOnGlobalLayoutListener(new OnGlobalLayoutListener() { 
@Override 
public void onGlobalLayout() { 
imageView.getViewTreeObserver().removeGlobalOnLayoutListener(this); 
textView.append("\n\n"+imageView.getHeight()+","+imageView.getWidth()); 
} 
});
现在要讨论的是当我们需要时候使用哪个方法呢?
现在把测试的Activity改成如下:


public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
final ImageView imageView = (ImageView) findViewById(R.id.imageview); 

//------------------------------------------------方法一 
int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED); 
int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED); 
imageView.measure(w, h); 
int height =imageView.getMeasuredHeight(); 
int width =imageView.getMeasuredWidth(); 
textView.append("\n"+height+","+width); 

System.out.println("执行完毕.."+System.currentTimeMillis()); 
}

接着来看下面几种方式输出结果:  
把测试Activity改成如下:

public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
final ImageView imageView = (ImageView) findViewById(R.id.imageview); 
-----------------------------------------------方法二 
ViewTreeObserver vto = imageView.getViewTreeObserver(); 
vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() { 
public boolean onPreDraw() { 
int height = imageView.getMeasuredHeight(); 
int width = imageView.getMeasuredWidth(); 
textView.append("\n"+height+","+width); 
return true; 
} 
}); 
}

结果如下:  


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

那么方法二和方法三在执行上有什么区别呢?
我们在布局文件中加入一个TextView来记录这个控件的宽高

<ScrollView 
android:layout_width="wrap_content" 
android:layout_height="wrap_content" > 

<TextView 
android:id="@+id/text" 
android:layout_width="wrap_content" 
android:layout_height="wrap_content" /> 
</ScrollView>

先来测试方法二:

public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
final ImageView imageView = (ImageView) findViewById(R.id.imageview); 
-----------------------------------------------方法二 
ViewTreeObserver vto = imageView.getViewTreeObserver(); 
vto.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() { 
public boolean onPreDraw() { 
int height = imageView.getMeasuredHeight(); 
int width = imageView.getMeasuredWidth(); 
textView.append("\n"+height+","+width); 
return true; 
} 
}); 
}


再来测试方法三

public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
final ImageView imageView = (ImageView) findViewById(R.id.imageview); 
//-----------------------------------------------方法三 
ViewTreeObserver vto2 = imageView.getViewTreeObserver(); 
vto2.addOnGlobalLayoutListener(new OnGlobalLayoutListener() { 
@Override 
public void onGlobalLayout() { 
imageView.getViewTreeObserver().removeGlobalOnLayoutListener(this); 
textView.append("\n\n"+imageView.getHeight()+","+imageView.getWidth()); 
} 
}); 
}




总结:那么需要获取控件的宽高该用那个方法呢?
方法一: 比其他的两个方法多了一次计算,也就是多调用了一次onMeasure()方法,该方法虽然看上去简单,但是如果要目标控件计算耗时比较大的话,不建议使用,如listView等.
方法二,它的回调方法会调用很多次,并且滑动TextView的时候任然会调用,所以不建议使用.
方法三,比较合适.
当然,实际应用的时候需要根据实际情况而定.


然而在实际应用中  我试了很多方法来获取view的高和宽

起初是延时几秒去获取长宽,但是这种方法治标不治本,具有一定程度的偶然性。

然后重写了view的父viewgroup的onLayout 、onMeasure 以及onDraw方法,在这些方法获取子view的宽度在通过handler传递回activity

但是不知为什么,传递的值有两种,且时对时不对,而且两个值整整差了50,使用以上的方法三亦会出现这个问题,对于这个问题在网上查询了很久也没有得到有用的信息,忘请大神们不吝赐教。

虽然没有找到问题的根本,也通过别的途径获取了真实值

方法如下:

如果想得到Width和Height,可以对View进行监听:

首先需要重写View的onSizeChange方法,加入监听回调函数:

public class MyTextView extends TextView {

	public MyTextView(Context context, AttributeSet attrs, int defStyle) {
		super(context, attrs, defStyle);
		// TODO Auto-generated constructor stub
	}

	public MyTextView(Context context, AttributeSet attrs) {
		super(context, attrs);
		// TODO Auto-generated constructor stub
	}

	public MyTextView(Context context) {
		super(context);
		// TODO Auto-generated constructor stub
	}

	SizeChangeListener l;

	public void setSizeChangeListener(SizeChangeListener orlExt) {
		l = orlExt;
	}

	@Override
	public void onSizeChanged(int w, int h, int oldw, int oldh) {
		// TODO Auto-generated method stub
		l.sizeChanged(w, h, oldw, oldh);
		super.onSizeChanged(w, h, oldw, oldh);
	}

	public interface SizeChangeListener {
		public void sizeChanged(int w, int h, int oldw, int oldh);
	}
}

然后在Activity中对View设置监听器:

tv_front = new MyTextView(this);
		tv_front.setBackgroundResource(R.drawable.slidebar);
		tv_front.setTextColor(Color.BLACK);
		tv_front.setText("要闻");
		tv_front.setGravity(Gravity.CENTER);
		tv_front.setTextSize(16);
		//监听textview尺寸改变
		tv_front.setSizeChangeListener(new SizeChangeListener() {
			
			@Override
			public void sizeChanged(int w, int h, int oldw, int oldh) {
				// TODO Auto-generated method stub
	                       // 这里能获取真实的高和宽
			}
		});

这样问题就圆满解决了。
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值