onCreate方法里面通过view.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" />  

测试的Activity的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,因为它自己还没有被测量,也就是说他自己都不知道自己有多高,而你这时候去获取它的尺寸,肯定是不行的。

解决方案一

使用view的measure方法
优点:可以立即获得宽和高
缺点:人为的多了一次测量过程

这种方法适用于需要在onCreate完成之前就获得一个view的宽和高的情况。
比如获得一个LinearLayout的宽和高

public int getViewWidth(LinearLayout view) {
        view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        return view.getMeasuredWidth();
    }
    public int getViewHeight(LinearLayout view) {
        view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        return view.getMeasuredHeight();
    }  

这种方法的原理是直接调用一个view或者viewgroup的measure方法去测量,测量之后该view的getMeasuredHeight()就会返回刚才测量所得的高,getMeasuredWidth返回测量所得宽。
本来在布局加载的过程中,view的measure方法一定会被系统调用(在onResume中已经调用了measure方法),但这发生在我们所不知道的某个时间点,为了在这之前提前得到测量结果,我们主动调用measure方法,但是这样做的好处是可以立即获得宽和高,坏处是多了一次测量过程。

至于为什么参数是LayoutParams.WRAP_CONTENT,那是因为我假设这个view的layout_width和layout_height为wrap_content,因为如果为一个确切的值,还有必要测量吗?

解决方案二

利用ViewTreeObserver的OnGlobalLayoutListener
优点:不需要额外的测量过程
缺点:只有在布局加载完成后,才能得到宽和高

OnGlobalLayoutListener是ViewTreeObserver的内部类,当一个视图树的布局发生改变时,可以被ViewTreeObserver监听到,这是一个注册监听视图树的观察者(observer),在视图树的全局事件改变时得到通知。ViewTreeObserver不能直接实例化,而是通过getViewTreeObserver()获得。
其中,我们可以利用OnGlobalLayoutListener来获得一个视图的真实高度,但是需要注意的是OnGlobalLayoutListener会被多次触发,因此在得到了高度之后,要将OnGlobalLayoutListener清除掉。

private void measureWidth() {
        //对控件layout结束事件进行监听(也不知道到底是先布局子空间还是父控件)
        view.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {// 当layout执行结束后回调
                //使用完必须撤销监听(只测量一次),否则,会一直不停的不定时的测量,这比较耗性能
                view.getViewTreeObserver().removeOnGlobalLayoutListener(this);//Added in API level 16
                //view.getViewTreeObserver().removeGlobalOnLayoutListener(this);//废弃了
                int width = view.getMeasuredWidth();
                int width2 = view.getWidth();//和上面的值是一样的
            }
        });
    }  

这种方法无法像第一种方法那样通过一个函数返回值,因为他是基于listener的,OnGlobalLayoutListener的onGlobalLayout被回调之前是没有值的。由于布局状态可能会发生多次改变,因此OnGlobalLayoutListener的onGlobalLayout可能被回调多次,所以我们在 第一次获得值之后就将listener注销掉。
**其他相关回调接口
**

ViewTreeObserver的内部回调接口:
OnGlobalLayoutListener    当在一个视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变时回调
OnPreDrawListener    当一个视图树将要绘制时,所要调用的回调函数的接口类
OnScrollChangedListener    当一个视图树中的一些组件发生滚动时回调
OnGlobalFocusChangeListener    当在一个视图树中的焦点状态发生改变时回调
OnTouchModeChangeListener    当一个视图树的触摸模式发生改变时回调
回调方法:
addOnGlobalFocusChangeListener 当在一个视图树中的焦点状态发生改变时调用
addOnGlobalLayoutListener 当在一个视图树中某个视图的可视状态发生改变时调用
addOnPreDrawListener 当一个视图树将要绘制时调用
addOnScrollChangedListener 当一个视图发生滚动时调用
addOnTouchModeChangeListener 当一个触摸模式发生改变时调用
dispatchOnGlobalLayout 当整个布局发生改变时通知相应的注册监听器
dispatchOnPreDraw 当一个视图树将要绘制时通知相应的注册监听器
isAlive 指示当前的ViewTreeObserver是否可用
removeGlobalOnLayoutListener 移除之前已经注册的全局布局回调函数
removeOnGlobalFocusChangeListener 移除之前已经注册的焦点改变回调函数
removeOnPreDrawListener 移除之前已经注册的预绘制回调函数
removeOnScrollChangedListener 移除之前已经注册的滚动改变回调函数
removeOnTouchModeChangeListener 移除之前已经注册的触摸模式改变回调函数
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容