如何在onCreate中测量View的实际宽高

通常在实际开发中,为了适配,我们会把View设置为match_parent或者wrap_content、又或者是设置weight权重来分配宽高,而不是使用具体值。那么就出现一个问题了,如果动态测量View的实际宽高。
其实这个问题网上有很多解决方法,无奈给出的方法众多,却没有解析,实际可能根本不管用。所以详细记录下这个问题。

测量宽高的API

我们以高度为例子:
android提供了两个API来动态获取View的高:

  • view.getMeasuredHeight()
  • view.getHeight()

那么这两个API的区别是什么呢?

简单来说就是:

  • view.getMeasuredHeight() 是由view中的测量方法赋值的,这个值包含了隐藏的高度(比如一个view部分超出屏幕,他也会计算出来)。
  • view.getHeight() 由view的底部位置减去顶部位置,即实际显示的View的高度,不包含隐藏了的高度。

如果是在View的onMeasure方法后执行上面两个方法,会发现可以得到正确的高度,但是如果是在onCreate等方法中执行就会发现返回的值不正确或者为0,这是为什么呢?

Activity和View的混合生命周期

我画了一个图来表明两者的生命周期先后问题:
混合生命周期
可以看到的是onCreate()和onResume()是在onMeasure方法之前。
说明此时还没有测量出实际的宽高,还没有进行绘制,所以调用上述的两个API会出现值为0或者数值错误的情况。

经过验证的解决方法

下面的方法都是进过验证的,但是有使用的条件限制,务必注意。

1.主动测量

代码:

view.measure(0, 0);  
view.getMeasuredWidth();
view.getMeasuredHeight();  

说明:
onMeasure传入的两个参数是由父控件的大小,
也可以使用 View.MeasureSpec.makeMeasureSpec(0,mode);设置值
其中mode可以选择

  • MeasureSpec.UNSPECIFIED 未指定尺寸,比如listview中尺寸由父控件决定
  • MeasureSpec.EXACTLY 适合match_parent或者具体值
  • MeasureSpec.AT_MOST 适合wrap_content不确定值

注意:
这种方法不一定能测出正确的值,因为onMesure会多次调用(由于onMesure自上而下,父控件如果对于子控件的宽高不满意,即如子控件没有限制宽高,父控件会重新调用onMesure重新测量),所以测量结果不一定正确。

2.使用OnPreDrawListener

代码:

ViewTreeObserver observer = vi
  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Android 连接 MySQL 数据库需要通过网络进行访问,因此我们需要在 AndroidManifest.xml 文件添加网络访问权限。请在 AndroidManifest.xml 文件添加以下代码: ```xml <uses-permission android:name="android.permission.INTERNET" /> ``` 接下来,我们可以在 onCreate() 方法使用 Java 的 JDBC(Java 数据库连接)技术连接 MySQL 数据库。以下是一个连接 MySQL 数据库的示例代码: ```java public class MainActivity extends AppCompatActivity { private static final String DB_URL = "jdbc:mysql://your_database_url:port/your_database_name"; private static final String USER = "your_username"; private static final String PASS = "your_password"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Connection conn = null; try { Class.forName("com.mysql.jdbc.Driver"); conn = DriverManager.getConnection(DB_URL, USER, PASS); System.out.println("Database connected successfully"); } catch (SQLException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } finally { try { if (conn != null) { conn.close(); } } catch (SQLException ex) { ex.printStackTrace(); } } } } ``` 在上述代码,我们使用了 com.mysql.jdbc.Driver 驱动程序连接 MySQL 数据库。请确保您已将 MySQL 驱动程序添加到项目。此外,您需要将 DB_URL、USER、PASS 替换为您的数据库 URL、用户名和密码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值