做安卓SDK如何获取当前Activity。

本文介绍在SDK开发中,当Application不可控且Activity由接入方创建时,如何通过反射ActivityThread类获取当前Activity实例的方法。详细展示了如何使用Class.forName、getMethod、invoke和getDeclaredField等技术来实现这一功能。

安卓开发中获取当前Activity我们平时一般采用如下两种方式

  • 通过ActivityLifecycleCallbacks接口监听当前的Activity实例,在onActivityResumed当中去获取。
  • 通过管理每一个Activity的onResume方法,记录所有的activity的状态。最后调用onResume的状态的Activity即为当前activity。

但做SDK的时候Application不是我们写的,其它的activity也可能是接入方开启的,这时我们就无法通过以上方式获取。此时可以采用反射系统的Activity管理类(ActivityThread)的方式获取。代码如下。

public static Activity getCurrentActivity () {
    try {
        Class activityThreadClass = Class.forName("android.app.ActivityThread");
        Object activityThread = activityThreadClass.getMethod("currentActivityThread").invoke(
                null);
        Field activitiesField = activityThreadClass.getDeclaredField("mActivities");
        activitiesField.setAccessible(true);
        Map activities = (Map) activitiesField.get(activityThread);
        for (Object activityRecord : activities.values()) {
            Class activityRecordClass = activityRecord.getClass();
            Field pausedField = activityRecordClass.getDeclaredField("paused");
            pausedField.setAccessible(true);
            if (!pausedField.getBoolean(activityRecord)) {
                Field activityField = activityRecordClass.getDeclaredField("activity");
                activityField.setAccessible(true);
                Activity activity = (Activity) activityField.get(activityRecord);
                return activity;
            }
        }
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    } catch (NoSuchFieldException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
    return null;
}
AndroidActivity 里,直接用 `getWidth()` 这类方法往往不能获取到 View 的宽度,像在 `onCreate`、`onStart`、`onResume` 这些生命周期方法中获取宽度,结果通常是 0。这是由于 View 的 measure 过程和 Activity 的生命周期并非同步执行,无法保证在这些方法执行时 View 已完成测量 [^1][^2][^3][^4]。 下面有几种可行的获取当前 Activity 中 View 宽度的方法: ### 重写 View 的 `onSizeChanged` 方法 在自定义 View 时重写 `onSizeChanged` 方法,当 View 的大小改变时会触发此方法,在这里可以获取到 View 的宽度。 ```java public class CustomView extends View { public CustomView(Context context) { super(context); } public CustomView(Context context, AttributeSet attrs) { super(context, attrs); } public CustomView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); int width = w; // 这里可以使用获取到的宽度 } } ``` ### 使用 `ViewTreeObserver` 通过 `ViewTreeObserver` 的 `addOnGlobalLayoutListener` 监听 View 树的全局布局变化,当布局完成后获取 View 的宽度。 ```java View view = findViewById(R.id.your_view_id); ViewTreeObserver viewTreeObserver = view.getViewTreeObserver(); viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { view.getViewTreeObserver().removeOnGlobalLayoutListener(this); } else { view.getViewTreeObserver().removeGlobalOnLayoutListener(this); } int width = view.getWidth(); // 这里可以使用获取到的宽度 } }); ``` ### 利用 `post` 方法 使用 `post` 方法将获取宽度的操作放到消息队列的末尾,确保在 View 测量完成后执行。 ```java View view = findViewById(R.id.your_view_id); view.post(new Runnable() { @Override public void run() { int width = view.getWidth(); // 这里可以使用获取到的宽度 } }); ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值