笔记:Activity 中 获取View 实际的宽/高

Question

在Activity 已启动的时候需要获取某个View 宽/高?

Analysis

通常操作会在onCreate或者onResume里面去获取这个View的宽/高,但是实际上无法正确获得某个View的宽/高的信息。这是因为View 的measure过程和Activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate、onResume时某个View已经测量完毕了,如果View还没有测量完毕了,那么获得 宽/高就是0.

Resolve
  • Activity/View.onWindowFocusChanged

该方法的含义是:View 已经初始化完毕了,宽/高已经准备好了,这个时候去获取宽/高是没问题的。注意: onWindowFocusChanged会被调用多次,当Activity 的窗口得到焦点和失去焦点时均会被调用一次。具体来说,如果频繁地进行onResume和onPause,那么onWindowFocusChanged也会频繁地调用。

public void onWindowFocusChanged(boolean hasFocus){
   super.onWindowFocusChanged(hasFocus);
   if(hasFocus){
     int width=view.getMeasuredWidth();
     int height=view.getMeasuredHeight();
   }
}
  • view.post(runnable) 推荐

通过post 可以将一个runnable投递到消息队列的尾部,然后等待Looper 调用此runnnable 的时候,View已经初始化好了。

view.post(
    new Runnable(){
    public void run(){
       int width=view.getMeasuredWidth();
       int height=vieew.getMeasuredHeight();
    }
  }
);
  • ViewTreeObserver

使用ViewTreeObserver 的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener 这个接口,当View 树的状态发生改变或者View 树内部的View的可见性发生改变时,onGlobalLayout 方法将被回调,因此这是获取View的宽/高一个很好的时机。注意: 伴随着View 树的状态改变等,onLayoutGlobal 会被调用多次。

ViewTreeObserver observer=view.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener(){
   public void onGlobalLayout(){
    view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
    int width=view.getMeasuredWidth();
    int height=vieew.getMeasuredHeight();    
   }
});
  • view.measure(int widthMeasureSpec, int heightMeasureSpec)

通过手动对View 进行measure来得到View 的宽/高。这里要分情况处理,根据View的LayoutParams来分:

  1. match_parent

这种情况无法measure出具体的高度。原因是:根据View 的measure过程,构造此种MeasureSpec 需要知道parentSize(父容器的剩余空间)。而这个时候我们无法知道parentSize 的大小,所以理论上不可能测量出View 的大小。

  1. 具体的数值(dp/px)

假设宽/ 高都是100px

int widthMeasureSpec=MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
int heightMeasureSpec=MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
view.measure(widthMeasureSpec,heightMeasureSpec);
  1. wrap_content
int widthMeasureSpec=MeasureSpec.makeMeasureSpec((1<<30-1),MeasureSpec.AT_MOST);
int heightMeasureSpec=MeasureSpec.makeMeasureSpec((1<<30-1),MeasureSpec.AT_MOST);
view.measure(widthMeasureSpec,heightMeasureSpec);

View 的尺寸使用30位二进制表示,也就是说最大是30个1(即2^30-1),也就是(1<<30)-1,在最大化模式下,用View 理论上能支持的最大值去构造MeasureSpec是合理的。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是一个基于Android的笔记添加标签的代码示例: 1. 定义布局文件: ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <EditText android:id="@+id/note_title" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="Note Title" /> <EditText android:id="@+id/note_content" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="Note Content" /> <EditText android:id="@+id/note_tags" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="Note Tags" /> <Button android:id="@+id/add_note_button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Add Note" /> </LinearLayout> ``` 2. 在Java代码获取布局的控件,并将标签作为字符串列表添加到笔记对象: ```java public class MainActivity extends AppCompatActivity { private EditText noteTitleEditText; private EditText noteContentEditText; private EditText noteTagsEditText; private Button addNoteButton; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); noteTitleEditText = findViewById(R.id.note_title); noteContentEditText = findViewById(R.id.note_content); noteTagsEditText = findViewById(R.id.note_tags); addNoteButton = findViewById(R.id.add_note_button); addNoteButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { String title = noteTitleEditText.getText().toString(); String content = noteContentEditText.getText().toString(); String tagsString = noteTagsEditText.getText().toString(); //将标签作为字符串列表添加到笔记对象 List<String> tags = Arrays.asList(tagsString.split(",")); Note note = new Note(title, content, tags); //TODO: 将笔记保存到数据库或其他存储位置 } }); } } ``` 以上代码示例仅供参考,具体实现可能因应用程序的需求而异。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值