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

来源:互联网 发布:淘宝卖家被处罚的原因 编辑:程序博客网 时间:2024/06/05 21:10

当我们试图在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()方法,该方法并不仅限于获取视图的宽和高。若读者有兴趣,可以更深入的去研究。

0 0
原创粉丝点击