天天看点

在onCreate()方法中获取View的宽度与高度

当我们试图在Activity的onCreate()方法中获取控件的宽和高时,遗憾的是如果我们在onCreate()方法中调用View的getHeight()和getWidth()方法,会发现返回值都是0。

为什么会是0呢?原来当onCreate()方法被调用时,会通过LayoutInflater将XML布局文件填充到ContentView。填充过程只包括创建视图,却不包括设置其大小。那么,视图的大小是在何时指定的呢?

Android开发文档的解释如下所示:

绘制布局由两个遍历过程组成:测量过程和布局过程。测量过程由measure(int, int)方法完成,该方法从上到下遍历视图树。在递归遍历过程中,每个视图都会向下层传递尺寸和规格。当measure方法遍历结束,每个视图都保存了各自的尺寸信息。第二个过程由layout(int, int, int, int)方法完成,该方法也是从上到下遍历视图树,在遍历过程中,每个父视图通过测量过程的结果定位所有子视图的位置信息。

结论如下:只有在整个布局绘制完毕后,视图才能得到自身的高和宽,这个过程发生在onCreate()方法之后,因此,在此之前调用getHeight()和getWidth()方法返回的结果都是0。

若把XML布局文件比喻成蛋糕食谱:LayoutInflater类就是购买所有食材的人;测量和布局的过程就是蛋糕师的工作,最终的视图就是蛋糕本身。在onCreate()阶段,只是购买了制作蛋糕的食材,但是仅仅知道食材是不足以预知蛋糕最终大小的。

我们可以使用View的post()方法解决上述问题。该方法接收一个Runnable线程参数,并将其添加到消息队列中,有趣的是Runnable线程会在UI线程中执行。

代码如下:

package com.example.huangfei.hack1;

import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {

    private TextView mTextView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mTextView = (TextView) findViewById(R.id.my_text_view);
        mTextView.post(new Runnable() {
            @Override
            public void run() {
                String size = String.format("TextView's width: %d, height: %d",
                        mTextView.getWidth(), mTextView.getHeight());
                Toast.makeText(MainActivity.this, size, Toast.LENGTH_SHORT).show();
            }
        });
    }

}
           
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="fill_parent"
              android:layout_height="fill_parent">

    <TextView android:id="@+id/my_text_view"
              android:layout_width="fill_parent"
              android:layout_height="wrap_content"
              android:gravity="center_horizontal"
              android:layout_gravity="center_vertical"
              android:text="Hello World, MainActivity!"/>

</LinearLayout>
           

Android源代码中很多模块都使用了post()方法,该方法并不仅限于获取视图的宽和高。若读者有兴趣,可以更深入的去研究。

继续阅读