关于获取当前Activity的一些思考

FEB 21ST, 2016

在Android开发过程中,我们有时候需要获取当前的Activity实例,比如弹出Dialog操作,必须要用到这个。关于如何实现由很多种思路,这其中有的简单,有的复杂,这里简单总结一下个人的一些经验吧。

反射

反射是我们经常会想到的方法,思路大概为

1 获取ActivityThread中所有的ActivityRecord 
2 从ActivityRecord中获取状态不是pause的Activity并返回

一个使用反射来实现的代码大致如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public static Activity getActivity() {    Class activityThreadClass = null;
    try {        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 (NoSuchMethodException e) {        e.printStackTrace();
    } catch (IllegalAccessException e) {        e.printStackTrace();
    } catch (InvocationTargetException e) {        e.printStackTrace();
    } catch (NoSuchFieldException e) {        e.printStackTrace();
    }
    return null;
}

然而这种方法并不是很推荐,主要是有以下的不足:

  • 反射通常会比较慢
  • 不稳定性,这个才是不推荐的原因,Android框架代码存在修改的可能性,谁要无法100%保证mActivitiespaused固定不变。所以可靠性不是完全可靠。

Activity基类

既然反射不是很可靠,那么有一种比较可靠的方式,就是使用Activity基类。

在Activity的onResume方法中,将当前的Activity实例保存到一个变量中。

1
2
3
4
5
6
7
8
public class BaseActivity extends Activity{
    @Override
    protected void onResume() {        super.onResume();
        MyActivityManager.getInstance().setCurrentActivity(this);
    }
}

然而,这一种方法也不仅完美,因为这种方法是基于约定的,所以必须每个Activity都继承BaseActivity,如果一旦出现没有继承BaseActivity的就可能有问题。

回调方法

介绍了上面两种不是尽善尽美的方法,这里实际上还是有一种更便捷的方法,那就是通过Framework提供的回调来实现。

Android自 API 14开始引入了一个方法,即Application的registerActivityLifecycleCallbacks方法,用来监听所有Activity的生命周期回调,比如onActivityCreated,onActivityResumed等。

So,一个简单的实现如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
public class MyApplication extends Application {

    @Override
    public void onCreate() {        super.onCreate();
        registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {            @Override
            public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
            }

            @Override
            public void onActivityStarted(Activity activity) {
            }

            @Override
            public void onActivityResumed(Activity activity) {                MyActivityManager.getInstance().setCurrentActivity(activity);
            }

            @Override
            public void onActivityPaused(Activity activity) {
            }

            @Override
            public void onActivityStopped(Activity activity) {
            }

            @Override
            public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
            }

            @Override
            public void onActivityDestroyed(Activity activity) {
            }
        });
    }
}

然而,金无足赤人无完人,这种方法唯一的遗憾就是只支持API 14即其以上。不过还在现在大多数设备都满足了这个要求。

为什么是弱引用

可能有人会带着疑问看到这里,MyActivityManager是个什么鬼,好,我们现在看一下这个类的实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class MyActivityManager {    private static MyActivityManager sInstance = new MyActivityManager();
    private WeakReference<Activity> sCurrentActivityWeakRef;

    private MyActivityManager() {
    }

    public static MyActivityManager getInstance() {        return sInstance;
    }

    public Activity getCurrentActivity() {        Activity currentActivity = null;
        if (sCurrentActivityWeakRef != null) {            currentActivity = sCurrentActivityWeakRef.get();
        }
        return currentActivity;
    }

    public void setCurrentActivity(Activity activity) {        sCurrentActivityWeakRef = new WeakReference<Activity>(activity);
    }

}

这个类,实现了当前Activity的设置和获取。

那么为什么要使用弱引用持有Activity实例呢?

其实最主要的目的就是避免内存泄露,因为使用默认的强引用会导致Activity实例无法释放,导致内存泄露的出现。详细了解弱引用,请参考本文译文:理解Java中的弱引用

转载自:http://droidyue.com/blog/2016/02/21/thinking-of-getting-the-current-activity-in-android/

原文有demo可以下载。

Android如何获得当前应用显示的Activity相关推荐

  1. Android应用程序中的多个Activity的显示创建和调用

    布局文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:too ...

  2. 关于android隐式启动activity的分析和说明,Android学习之Intent中显示意图和隐式意图的用法实例分析...

    本文实例讲述了Android学习之Intent中显示意图和隐式意图的用法.分享给大家供大家参考,具体如下: Intent(意图)主要是解决Android应用的各项组件之间的通讯. Intent负责对应 ...

  3. android应用去掉状态栏_Android 显示、隐藏状态栏和导航栏

    Android 显示.隐藏状态栏和导航栏 控制状态栏显示,Activity的主题中配置全屏属性 true 控制状态栏显示,在setContentView之前设置全屏的flag getWindow(). ...

  4. Android -- TextView与EditText 同步显示

    Android -- TextView与EditText 同步显示 文章分类:JavaEye 方法一.利用View.OnKeyListener"同步"显示         Java ...

  5. android 弹出对话框时显示键盘

    今天,简单讲讲如何在弹出自定义的对话框时,弹出软键盘. 之前,我讲了如何制作自定义的对话框,现在,如果在弹出对话框时,自动弹出软键盘,将光标直接聚焦到自定义对话框的输入框内,那该怎么做呢? 在网上搜索 ...

  6. Android之解析GML并显示

    本例主要实现在APP中解析GML数据并显示 GML,地理标记语言(外语全称:Geography MarkupLanguage.外语缩写:GML),它由开放式地理信息系统协会(外语缩写:OGC)于199 ...

  7. android 根据滑动隐藏或显示导航 类似手机QQ好友个人信息

    //重写ScrollViewpublic class NotifyingScrollView extends ScrollView {/*** @author Cyril Mottier*/publi ...

  8. android activity从新打开,【Android开发-8】生命周期,Activity中打开另一个Activity

    前言:生命中有不少人陪伴本身走过一辈子中的某段旅程,仅仅是有些人仅仅是某阶段出现,有些人却陪伴本身很是久.就像小学.中学.高中.大学,那些之前觉得会长久拥有的,当经历过天涯各地地忙碌于生活,或如意.或 ...

  9. Android基础新手教程——4.1.3 Activity登堂入室

    Android基础新手教程--4.1.3 Activity登堂入室 标签(空格分隔): Android基础新手教程 本节引言: 好的,在学习了两节的Activity后相信大家已经知道怎样去使用Acti ...

最新文章

  1. 24个必须掌握的数据库面试问题~
  2. R语言-文本挖掘 主题模型 文本分类
  3. html图片上下左右滑动,一个支持任意尺寸的图片上下左右滑动效果
  4. QT中关于ipv6和getaddressinfo的开关
  5. mysql bypass_Bypass MySQL Safedog
  6. linux下无root权限使用yum安装的方法
  7. IOS内购流程从0-1手把手教会
  8. TortoiseGit 本地仓库和远程仓库建立联系_入门试炼_02
  9. 支持高并发的IIS Web服务器常用设置 II
  10. l380废墨收集垫已到使用寿命_湖北土工网垫
  11. Bagging Classifier+Regressor
  12. 安装linux可是c盘文件夹失败,硬盘安装linux失败,进不去以前的windows怎么处理?
  13. 基于FPGA----VGA显示跳动的小白框设计
  14. springboot+mybatis+druid 多数据源整合
  15. 分页内存与非分页内存导致的蓝屏死机问题
  16. 【毕设论文——必修篇】开题报告要写些什么?这里有参考模板
  17. 针对传感网的数据管理系统结构有_2010年自考管理信息系统模拟试题及答案(三)...
  18. C语言画直线~Bresenham方法
  19. From Nand to Tetris Week1 超详细2021
  20. 计算机快捷键里面不显示桌面,桌面显示,显示桌面快捷键不见了

热门文章

  1. css3魔方3乘3每层旋转_在玩魔方中学数学,原来魔方与矩阵还有这样的关系
  2. linux实现乘法函数,linux命令行计算器
  3. apache添加支持php的模块,配置Apache支持PHP5 apache php 套件 apache添加php模块 apache部署php项...
  4. 在java中关于枚举类型的特性_java枚举类型小结
  5. 计算机程程序员英语,计算机程序员常用英语词汇(2)
  6. css画个框,用CSS绘制带有边框的尖端
  7. python中的字典推导式_17.python 字典推导式(经典代码)
  8. 农艺师需要职称计算机,2015年农艺师职称计算机考试宝典.doc
  9. python怎么设置界面的背景音乐_PyQt5图形界面播放音乐的实例
  10. 《密码与安全新技术专题》第11周作业