`
zzu_007
  • 浏览: 23752 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Acitivy创建Context的过程(一)

阅读更多
page1
从本篇文章开始,我们分析一下Activity创建Context的过程.
Context是在ActivityThread的performLaunchActivity函数中创建的, 因此我们就从performLaunchActivity函数作为入口开始分析:
1     private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
2         // System.out.println("##### [" + System.currentTimeMillis() + "] ActivityThread.performLaunchActivity(" + r + ")");
3
4         ActivityInfo aInfo = r.activityInfo;
5         if (r.packageInfo == null) {
6             r.packageInfo = getPackageInfo(aInfo.applicationInfo, r.compatInfo,
7                     Context.CONTEXT_INCLUDE_CODE);
8         }
9
10         ComponentName component = r.intent.getComponent();
11         if (component == null) {
12             component = r.intent.resolveActivity(
13                 mInitialApplication.getPackageManager());
14             r.intent.setComponent(component);
15         }
16
17         if (r.activityInfo.targetActivity != null) {
18             component = new ComponentName(r.activityInfo.packageName,
19                     r.activityInfo.targetActivity);
20         }
21
22         Activity activity = null;
23         try {
24             java.lang.ClassLoader cl = r.packageInfo.getClassLoader();
25             activity = mInstrumentation.newActivity(
26                     cl, component.getClassName(), r.intent);
27             StrictMode.incrementExpectedActivityCount(activity.getClass());
28             r.intent.setExtrasClassLoader(cl);
29             if (r.state != null) {
30                 r.state.setClassLoader(cl);
31             }
32         } catch (Exception e) {
33             if (!mInstrumentation.onException(activity, e)) {
34                 throw new RuntimeException(
35                     "Unable to instantiate activity " + component
36                     + ": " + e.toString(), e);
37             }
38         }
39
40         try {
41             Application app = r.packageInfo.makeApplication(false, mInstrumentation);
42
43             if (localLOGV) Slog.v(TAG, "Performing launch of " + r);
44             if (localLOGV) Slog.v(
45                     TAG, r + ": app=" + app
46                     + ", appName=" + app.getPackageName()
47                     + ", pkg=" + r.packageInfo.getPackageName()
48                     + ", comp=" + r.intent.getComponent().toShortString()
49                     + ", dir=" + r.packageInfo.getAppDir());
50
51             if (activity != null) {
52                 Context appContext = createBaseContextForActivity(r, activity);
53                 CharSequence title = r.activityInfo.loadLabel(appContext.getPackageManager());
54                 Configuration config = new Configuration(mCompatConfiguration);
55                 if (DEBUG_CONFIGURATION) Slog.v(TAG, "Launching activity "
56                         + r.activityInfo.name + " with config " + config);
57                 activity.attach(appContext, this, getInstrumentation(), r.token,
58                         r.ident, app, r.intent, r.activityInfo, title, r.parent,
59                         r.embeddedID, r.lastNonConfigurationInstances, config);
60
61                 if (customIntent != null) {
62                     activity.mIntent = customIntent;
63                 }
64                 r.lastNonConfigurationInstances = null;
65                 activity.mStartedActivity = false;
66                 int theme = r.activityInfo.getThemeResource();
67                 if (theme != 0) {
68                     activity.setTheme(theme);
69                 }
70
71                 activity.mCalled = false;
72                 mInstrumentation.callActivityOnCreate(activity, r.state);
73                 if (!activity.mCalled) {
74                     throw new SuperNotCalledException(
75                         "Activity " + r.intent.getComponent().toShortString() +
76                         " did not call through to super.onCreate()");
77                 }
78                 r.activity = activity;
79                 r.stopped = true;
80                 if (!r.activity.mFinished) {
81                     activity.performStart();
82                     r.stopped = false;
83                 }
84                 if (!r.activity.mFinished) {
85                     if (r.state != null) {
86                         mInstrumentation.callActivityOnRestoreInstanceState(activity, r.state);
87                     }
88                 }
89                 if (!r.activity.mFinished) {
90                     activity.mCalled = false;
91                     mInstrumentation.callActivityOnPostCreate(activity, r.state);
92                     if (!activity.mCalled) {
93                         throw new SuperNotCalledException(
94                             "Activity " + r.intent.getComponent().toShortString() +
95                             " did not call through to super.onPostCreate()");
96                     }
97                 }
98             }
99             r.paused = true;
100
101             mActivities.put(r.token, r);
102
103         } catch (SuperNotCalledException e) {
104             throw e;
105
106         } catch (Exception e) {
107             if (!mInstrumentation.onException(activity, e)) {
108                 throw new RuntimeException(
109                     "Unable to start activity " + component
110                     + ": " + e.toString(), e);
111             }
112         }
113
114         return activity;
115     }
第52行(ActivityThread->performLaunchActivity)调用createBaseContextForActivity函数为刚刚创建的activity对象创建Context对象, 关于performLaunchActivity函数的详细分析可以参考page2文件.
第57-59行(ActivityThread->performLaunchActivity)调用Activity的attach函数, 关于attach函数的详细分析可以参考page5文件.
page2
ActivityThread的createBaseContextForActivity函数定义如下:
1     private Context createBaseContextForActivity(ActivityClientRecord r,
2             final Activity activity) {
3         ContextImpl appContext = new ContextImpl();
4         appContext.init(r.packageInfo, r.token, this);
5         appContext.setOuterContext(activity);
6
7         // For debugging purposes, if the activity's package name contains the value of
8         // the "debug.use-second-display" system property as a substring, then show
9         // its content on a secondary display if there is one.
10         Context baseContext = appContext;
11         String pkgName = SystemProperties.get("debug.second-display.pkg");
12         if (pkgName != null && !pkgName.isEmpty()
13                 && r.packageInfo.mPackageName.contains(pkgName)) {
14             DisplayManagerGlobal dm = DisplayManagerGlobal.getInstance();
15             for (int displayId : dm.getDisplayIds()) {
16                 if (displayId != Display.DEFAULT_DISPLAY) {
17                     Display display = dm.getRealDisplay(displayId);
18                     baseContext = appContext.createDisplayContext(display);
19                     break;
20                 }
21             }
22         }
23         return baseContext;
24     }
第3行(ActivityThread->createBaseContextForActivity)会new一个ContextImpl对象. ContextImpl的构造函数的详细分析可以参考page3文件.
第4行(ActivityThread->createBaseContextForActivity)会调用ContextImpl的init函数, 关于init函数的详细分析可以参考page4文件.
第5行(ActivityThread->createBaseContextForActivity)调用ContextImpl的setOuterContext函数, ContextImpl的setOuterContext函数定义如下:
final void setOuterContext(Context context) {
        mOuterContext = context;
    }
这样, ContextImpl也会拿着activity对象.

第10-22行(ActivityThread->createBaseContextForActivity)是干什么呢?不知道.
page3
在这里我们分析一下ContextImpl类的构造过程.我们先来看一下ContextImpl类的继承体系, ContextImpl类的定义如下:
class ContextImpl extends Context {
public abstract class Context {

ContextImpl类的构造函数如下所示
ContextImpl() {
        mOuterContext = this;
    }
在ContextImpl的构造函数中, 只是初始化了成员变量mOuterContext, 使之指向该ContextImpl对象.
成员变量mOuterContext的定义如下:
private Context mOuterContext;

妈的, 就这么简单.
分享到:
评论

相关推荐

    Android启动APP时黑屏白屏的解决方法

    首先系统的服务会先检查startActivity中的intent的信息,然后在去创建进程,最后才是执行启动Acitivy的操作。而我们上面提到的显示白黑屏的问题,就是在这段时间内产生的。 系统在绘制页面加载布局之前,首先会初始...

    Android Service中使用Toast无法正常显示问题的解决方法

    在做Service简单练习时,在Service中的OnCreate、OnStart、OnDestroy三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法: DemoService...

    数学建模拟合与插值.ppt

    数学建模拟合与插值.ppt

    [net毕业设计]ASP.NET教育报表管理系统-权限管理模块(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    mysql相关资源.txt

    mysql相关资源.txt

    利用HTML+CSS+JS的国漫分享网站(响应式)

    此项目为一个HTML+CSS+JS的国漫分享网站,用户可以在此网站中观看自己喜欢的国漫。此网站共有4个页面,分别为首页,最新动态,热门推荐,分类。页面动漫图片齐全,内容可更改。可用于期末课程设计或个人课程设计。

    Python爬虫爬取漫画

    Python爬虫爬取漫画

    C++语言编程用模拟退火算法解决旅行商问题

    模拟退火算法应用。C++语言编程用模拟退火算法解决旅行商问题。该资源包含模拟退火算法C++语言的源代码。模拟退火算法是一种基于概率的全局优化算法,最初来自于物理学中的退火过程。它通过模拟金属冷却时原子排列逐渐趋于最低能量状态的过程来寻找问题的最优解。模拟退火算法常用于解决非线性、组合优化问题,特别适合于大规模、复杂的搜索空间。

    传感器试题及答案.doc

    传感器试题及答案.doc

    [net毕业设计]ASP.NET网上书店(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    MongoDB数据表基本操作中文最新版本

    本文档主要讲述的是MongoDB数据表基本操作;希望对大家会有帮助;感兴趣的朋友可以过来看看

    1-全国各省废气、废水排放二氧化硫、氮氧化物、烟尘、颗粒物排放量统计数据2011-2021年-社科数据.zip

    本数据集提供了2011至2021年间全国各省废气和废水中主要污染物的排放量统计数据。数据涵盖了二氧化硫、氮氧化物、烟尘和颗粒物等关键污染物的排放量,为研究中国环境状况和污染物排放趋势提供了宝贵信息。数据显示,2011-2021年间,各省的二氧化硫排放量从数十万吨到数百万吨不等,其中广东、广西、海南等省份的排放量较高。氮氧化物排放量同样显示出地域差异,北京、天津等北方城市的排放量相对较低,而一些工业大省如河北、山西的排放量较高。颗粒物排放量统计显示,工业源和生活源是主要的排放源,其中工业源排放量占比较大。这些数据不仅对环保政策制定者具有参考价值,也为学术研究提供了实证基础。

    脉冲宽度测量单片机课程设计.doc

    脉冲宽度测量单片机课程设计.doc

    [net毕业设计]ASP.NET在线毕业论文提交系统的设计与实现(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    求职与招聘(源代码+论文+说明文档).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    [net毕业设计]ASP.NET视频点播系统的设计与实现(源代码+论文).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    全国矢量地图数据【国道+高速】-ArcGis Shape 格式数据集

    全国矢量地图数据【国道+高速公路】ArcGIS Shape格式数据集是一种专门用于地理信息系统(GIS)的矢量数据集,包含中国范围内国道和高速公路的详细路网信息。该数据集广泛应用于交通规划、导航、物流分析和灾害应急等领域,具有高精度和易用性。 数据集特点: 1. 数据内容: 国道:包括以“G”开头的国家级公路,如G1京哈高速、G107国道等。 高速公路:包括全国范围内的所有高速公路网,覆盖主要经济区、城市和边境口岸。 属性数据: 道路编号(国道或高速公路编号)。 道路名称。 道路等级(如一级、二级、快速路等)。 起点和终点坐标。 道路长度(单位:公里)。 相关属性(如路段建成年份、设计速度、车道数等)。 2. 数据格式: **Shapefile(.shp)**格式,支持主流GIS软件(如ArcGIS、QGIS)及数据处理工具(如Python、Matlab)。 3. 投影坐标系: 一般采用WGS84地理坐标系,或可根据需求转换为**GCJ-02(火星坐标系)**以配合国内导航应用。

    4.html

    4

    ASP网上作业提交系统(源代码+论文)(源代码+论文+说明文档).zip

    【项目资源】:包含前端、后端、移动开发、操作系统、人工智能、物联网、信息化管理、数据库、硬件开发、大数据、课程资源、音视频、网站开发等各种技术项目的源码。包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。【项目质量】:所有源码都经过严格测试,可以直接运行。功能在确认正常工作后才上传。【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。【附加价值】:项目具有较高的学习借鉴价值,也可直接拿来修改复刻。对于有一定基础或热衷于研究的人来说,可以在这些基础代码上进行修改和扩展,实现其他功能。【沟通交流】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。鼓励下载和使用,并欢迎大家互相学习,共同进步。

    Oracle常用分析函数说明中文最新版本

    Oracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是对于每个组返回多行,而聚合函数对于每个组只返回一行。 感兴趣的朋友可以过来看看

Global site tag (gtag.js) - Google Analytics