1、View的getWidth()和getMeasuredWidth()有什么区别吗?
getMeasuredWidth()获取的是view的原始大小,也就是这个view在XML文件中配置或者是代码中设置的大小。getWidth()获取的是这个view最终显示的大小,这个大小可能与原始大小一致也有可能不一致。
/**
* Like {@link #getMeasuredWidthAndState()}, but only returns the
* raw width component (that is the result is masked by
* {@link #MEASURED_SIZE_MASK}).
* @return The raw measured width of this view.
*/
public final int getMeasuredWidth() {
return mMeasuredWidth & MEASURED_SIZE_MASK;
}
getMeasuredWidth()获取的是mMeasuredWidth的这个值。这个值是一个8位的十六进制的数字,高两位表示的是这个measure阶段的Mode的值,具体可以查看MeasureSpec的原理。这里mMeasuredWidth & MEASURED_SIZE_MASK表示的是测量阶段结束之后,view真实的值。而且这个值会在调用了setMeasuredDimensionRaw()函数之后会被设置。所以getMeasuredWidth()的值是measure阶段结束之后得到的view的原始的值。
/**
* Return the width of the your view.
* @return The width of your view, in pixels.
*/
@ViewDebug.ExportedProperty(category = "layout")
public final int getWidth() {
return mRight - mLeft;
}
从源码可以看出,getwidth返回的是右边坐标减轻坐标减去左边坐标,这要在布局之后才能确定它们的坐标,也就是说在布局后才能调用getwidth来获取。所以getWidth()获得的宽度是View在设定好布局后整个View的宽度。
2、如何在onCreate中拿到View的宽度和高度?
如果需要开发一些需要依赖UI控件的宽度和高度的功能,这时候我们需要用到view的宽高,在实际开发过程中我们会发现,直接在onCreate中获取view的宽高值为0,原因在于当onCreate()方法被调用的时候会通过LayoutInflater将xml文件填充到ContentView。填充过程中只包括创建视图,不包括设置视图大小。我们可以通过以下几种方法来获取view的宽高。
①、在onWindowFocusChanged方法中取到view的宽高,该方法的含义就是view已经初始化完毕,并且该方法在Activity窗口得到和是去焦点时均会调用。
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
Log.d(TAG,"width:" + mLeftBtn.getWidth());
Log.d(TAG,"height:" + mLeftBtn.getHeight());
}
12-16 10:58:07.930 3980-3980/com.ys.startest D/FifthActivity: width:960
12-16 10:58:07.930 3980-3980/com.ys.startest D/FifthActivity: height:48
②、view.post(runnable)
通过post可以将一个runnable投递到消息队列的尾部,然后等待UI线程Looper调用此runnable的时候,view也已经初始化好了。
mLeftBtn.post(new Runnable() {
@Override
public void run() {
Log.d(TAG,"view.post width:" + mLeftBtn.getWidth());
Log.d(TAG,"view.post height:" + mLeftBtn.getHeight());
}
});
12-16 11:00:18.580 4156-4156/com.ys.startest D/FifthActivity: view.post width:960
12-16 11:00:18.580 4156-4156/com.ys.startest D/FifthActivity: view.post height:48
③、ViewTreeObserver
使用ViewTreeObserver的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener这个接口,当view树的状态发生改变或者view树内部的view的可见性发生改变时,onGlobalLayout方法将被回调,因此这是获取view的宽高一个很好的时机。需要注意的是,伴随着view树的状态改变等,onGlobalLayout会被调用多次。
ViewTreeObserver viewTreeObserver = mLeftBtn.getViewTreeObserver();
viewTreeObserver.addOnGlobalLayoutListener(
new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
Log.d(TAG, "view.post width:" + mLeftBtn.getWidth());
Log.d(TAG, "view.post height:" + mLeftBtn.getHeight());
}
}
);
12-16 11:12:45.800 4861-4861/com.ys.startest D/FifthActivity: view.post width:960
12-16 11:12:45.800 4861-4861/com.ys.startest D/FifthActivity: view.post height:48
12-16 11:12:45.840 4861-4861/com.ys.startest D/FifthActivity: view.post width:960
12-16 11:12:45.840 4861-4861/com.ys.startest D/FifthActivity: view.post height:48