LinearLayout获取动态的高度+ 面试题讲解(图文并茂)

一、直接获取控件的高度,因为在onCreate方法的LinearLayout还并没有绘制完成,所以获取的高度为0,或者试着把这段代码放到onResume()方法中,依然还是0。

public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  LinearLayout linearlayout = (LinearLayout) findViewById(R.id.layoutInfo);
 Log.i("jhq", linearlayout .getWidth()+"L"+linearlayout .getHeight());
}

二、那如何解决这个问题:下面有两方法:

1、可以通过定时器不断的监听LinearLayout的宽高,等绘制完成后关闭定时器即可

//可以通过定时器不断的监听LinearLayout的宽高,等绘制完成后关闭定时器即可
final Handler handler= new Handler(){
   @Override
   public void handleMessage(Message msg) {
   if(msg.what == 1) {
    if(ll.getWidth()!=0) {
    Log.i("w", ll.getWidth()+"L"+ll.getHeight());
      timer.cancel();
 
    }
   } 
   }
  };
  timer = new Timer();
  TimerTask task = new TimerTask(){
   public void run() { 
    Message message = new Message(); 
    message.what = 1; 
    myHandler.sendMessage(message); 
    } 
   }; 
  timer.schedule(task,10,1000); 
 }


======================================延申==========================================
类似,如果想在Activity启动后立即弹出PopupWindow,我们知道在Activity的onCreate()方法中直接写弹出PopupWindow方法会报错,因为activity没有完全启动是不能弹出PopupWindow。
我们可以尝试用两种方法实现:

1、用onWindowFocusChanged方法
@Override
public void onWindowFocusChanged(boolean hasFocus) {
 super.onWindowFocusChanged(hasFocus);
 showPopupWindow();
}


2、用Handler和Runnable,延时
@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);
 mHandler.postDelayed(mRunnable, 1000);
}
private Runnable mRunnable = new Runnable() {
 public void run() {
 showPopupWindow();
 }
};

2、view.post(Runnable)

//直接在onCreate()写下面的
linearlayout.post(new Runnable() {
            @Override
            public void run() {
                linearlayout.getHeight();
                Log.d(TAG, "run: "+linearlayout.getHeight());
            }
        });

3、先测量在获取测量的宽度、高度

由于Android程序的运行机制决定了无法再组件类外部使用getWidth和getHeight方法获得高度和宽度(在自定义组件类中可以实现),必须使用View.getMeasuredWidth和View.getMeasureHeight方法获得当前组件的宽度和高度,在调用这两个方法之前,必须调用View.measure方法先测量组件宽度和高度。
如果想直接获取在布局文件中定义的组件的宽度和高度,可以直接使用View.getLayoutParams().width和View.getLayoutParams().height

代码如下:

View view = getLayoutInflater().inflate(R.layout.activity_main, null);  
LinearLayout linearlayout = (LinearLayout)view.findViewById(R.id.linearlayout);  
//measure方法的参数值都设为0即可  
linearlayout.measure(0,0);  
//获取组件宽度  
int width = linearlayout.getMeasuredWidth();  
//获取组件高度  
int height = linearlayout.getMeasuredHeight();  

面试题:如何获得LinearLayout的宽度和高度?

解析:此题表面看来是关于LinearLayout的,实际上也可以扩展到其他继承于View组件中。其他编程语言使用View.getWidth和View.getHeight方法获得宽度和高度在Android程序中返回的都是0.由于Android程序的运行机制决定了无法在组件类外部使用getWidth和getHeight方法获得高度和宽度(在自定义组件类的内容可以通过这两个方法获取当前组件的宽度和高度,如在onSizeChanged方法中可以使用这两个方法获得宽度和高度),必须使用View.getMeasuredWidth和View.getMeasuredHeight方法获取当前组件的宽度和高度。在调用这两个方法之前,必须调用View.measure方法先测量组件的宽度和高度。

如果想直接获取在布局文件中定义的组件的宽度和高度,可以直接使用View.getLayoutParams().width和View.getLayoutParams().height。当宽度和高度被设为fill_parent、match_parent、wrap_content时,这两个变量会返回MATCH_PARENT(-1)、FILL_PARENT(-1)、WRAP_CONTENT(-2)常量的值。

答案:由于LinearLayout是View的子类,因此,可以使用View.getMeasuredWidth和View.getMeasuredHeight方法获取当前组件的宽度和高度,代码如下:

View view = getLayoutInflater().inflate(R.layout.main, null);
LinearLayout linearlayout = (LinearLayout)view.findViewById(R.id.linearlayout);
//measure方法的参数值都设为0即可
linearlayout.measure(0,0);
//获取组件的宽度
int width = linearlayout.getMeasuredWidth();
//获取组件的高度
int width = linearlayout.getMeasuredHeight();

注意:如果组件的宽度或高度设为fill_parent或match_parent,使用View.getMeasuredWidth和View.getMeasuredHeight方法获取组件宽度或高度时,当组件中包含其他子组件时,所获得的的实际值是这些组件所占的最小宽度和最小高度。

[来理解上面这句话,就会引申出一个问题的考虑{LinearLayout的高度到底由谁决定?}看下面的xml的显示效果:绿色是最外曾LinearLayout的背景色,红色是第四个子view-LinearLayout的背景色]

//图1
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/root"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="#aaFF99"
    android:orientation="horizontal">

    <TextView
        android:id="@+id/tv1"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:text="高度"
        android:textSize="10sp" />

    <TextView
        android:id="@+id/tv2"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:text="高度"
        android:textSize="20sp" />

    <TextView
        android:id="@+id/tv3"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:text="高度"
        android:textSize="40sp" />

    <LinearLayout
        android:id="@+id/linearLayout"
        android:layout_width="0dp"
        android:layout_height="20dp"
        android:layout_weight="1"
        android:background="@color/colorAccent"
        android:minHeight="100dp">

    </LinearLayout>

</LinearLayout>

图1-效果(结论:父LinearLayout的高度,是取的子LinearLayout的高度20dp,而不是minHeight=100dp)

修改xml:只修改第四个子view-LinearLayout的布局如下:

<LinearLayout
        android:id="@+id/linearLayout"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:background="@color/colorAccent"
        android:minHeight="100dp">

 </LinearLayout>

 图2:(结论:父LinearLayout的高度,是取的子LinearLayout的minHeight=100dp,此时layout_height=wrap_content)

总结

有时候,源码真的很重要,被自己的常识限制了,以为如果都是match_parent,就真的match_parent了,仔细想一想真的很有趣,LinearLayout就想一个父亲,高度为包裹内容,他有四个孩子,其中三个孩子高度都是match_parent,就是说,父亲,我和你一样高就够了,其中一个孩子说,我不,我的高度是包裹自己的内容,父亲就说,好的,我们都不知道自己的高度,只有你知道自己的高度,那么我们就都和你一样高吧!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值