Android世界第一个activity启动过程

第一次使用Markdown,感觉不错。

Android系统从按下开机键一直到launcher的出现,是一个如何的过程,中间都做出了什么操作呢。带着这些疑问開始源代码之旅。

像windows操作系统一样,每一个系统的启动都会有一个引导程序,在linux中,当引导程序启动linux内核后,会载入各种驱动和数据结构。当有了驱动之后。開始载入Android系统,開始进入linux世界的第一个进程:init进程。

在init.c的main中:

    int main(int argc, char **argv){
umask(0);// 清除文件的默认属性
mkdir("/dev", 0755); // 创建文件、挂载文件等操作
........
init_parse_config_file("/init.rc"); // 解析文件
.........
}

在init.rc文件里:(该文件在system/core/rootdir文件夹下)

    // 设置一些全局环境变量
export PATH /sbin:/vendor/bin:/system/sbin:/system/bin:/system/xbin
export LD_LIBRARY_PATH /vendor/lib:/system/lib
..............
// 创建主要的文件系统结构
mkdir /data/misc 01771 system misc
..............
// 启动一些服务
service zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server
socket zygote stream 666
onrestart write /sys/android_power/request_state wake
onrestart write /sys/power/state on
onrestart restart media
onrestart restart netd
.............

最重要的是这个zygote进程。zygote就是一个孵化器,相似于母进程一样,能够fork出非常多的子进程。是Android的一个母进程,用来启动Android的其它服务进程。当media、netd等服务进程销毁后。zygote进程会自己主动重新启动这些服务进程

在App_Main.cpp文件里:

    int main(int argc, const char* const argv[]){
............................
bool startSystemServer = (i < argc) ?
strcmp(argv[i], "--start-system-server") == 0 : false;
setArgv0(argv0, "zygote");
set_process_name("zygote");
runtime.start("com.android.internal.os.ZygoteInit",startSystemServer);
}

在AndroidRuntime的start方法中

    void AndroidRuntime::start(const char* className, const bool startSystemServer){
....................
// 开启java虚拟机,并载入好jni执行环境
if (startVm(&mJavaVM, &env) != 0)
goto bail;
.............
// 利用jni与java进行交互,载入ZygoteInit类
startClass = env->FindClass(slashClassName);
if (startClass == NULL) {
LOGE("JavaVM unable to locate class '%s'\n", slashClassName);
} else {
// 利用jni调用ZygoteInit类中的main方法
startMeth = env->GetStaticMethodID(startClass, "main","([Ljava/lang/String;)V");
if (startMeth == NULL) {
LOGE("JavaVM unable to find main() in '%s'\n", className);
} else {
env->CallStaticVoidMethod(startClass, startMeth, strArray);
if (env->ExceptionCheck())
threadExitUncaughtException(env);
}
}

在ZygoteInit.java中:

    public static void main(String argv[]) {
// 设置Android执行时的最小堆大小5M
VMRuntime.getRuntime().setMinimumHeapSize(5 * 1024 * 1024);
..............
// 预载入一些经常使用的类。这些经常使用的类在2.3中有1800个左右,在4.2源代码中大概有2400多个经常使用的
// 像有些手机厂商手机的启动速度较快的,预计是对这里进行了优化
preloadClasses();
// 载入一些资源文件。array、drawable、color等xml文件
preloadResources();
..............
if (argv[1].equals("true")) {
//在SystemServer类中fork系统服务进程
startSystemServer();
} else if (!argv[1].equals("false")) {
throw new RuntimeException(argv[0] + USAGE_STRING);
}
}
private static boolean startSystemServer() throws MethodAndArgsCaller, RuntimeException {
String args[];
String ashmem_size = System.getProperty("gralloc.ashmem_size");
if ((null != ashmem_size) && (0 != ashmem_size.length())) {
args = new String[] {
"--setuid=1000",
"--setgid=1000",
"--setgroups=1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,1018,3001,3002,3003,3006",
"--capabilities=130104352,130104352",
"--rlimit=8,",
"--runtime-init",
"--nice-name=system_server",
"com.android.server.SystemServer",
};
args[4] = args[4].concat(ashmem_size);
args[4] = args[4].concat(",");
args[4] = args[4].concat(ashmem_size);
} else {
args = new String[] {
"--setuid=1000",
"--setgid=1000",
"--setgroups=1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,1018,3001,3002,3003,3006",
"--capabilities=130104352,130104352",
"--runtime-init",
"--nice-name=system_server",
"com.android.server.SystemServer",
};
}
..............................
pid = Zygote.forkSystemServer(
parsedArgs.uid, parsedArgs.gid,
parsedArgs.gids, debugFlags, rlimits,
parsedArgs.permittedCapabilities,
parsedArgs.effectiveCapabilities);
} catch (IllegalArgumentException ex) {
throw new RuntimeException(ex);
}

SystemServer类中:

     native public static void init1(String[] args);
.........
System.loadLibrary("android_servers");
init1(args);
.........
}

首先载入android_servers这个so库,这个库在于systemServer父文件夹同级别下的jni文件夹中。相应的c文件是com_android_server_SystemServer.c。然后调用库中的init1方法,

     extern "C" int system_init();
static void android_server_SystemServer_init1(JNIEnv* env, jobject clazz)
{
system_init();
}
static JNINativeMethod gMethods[] = {
{ "init1", "([Ljava/lang/String;)V", (void*) android_server_SystemServer_init1 },
};
int register_android_server_SystemServer(JNIEnv* env)
{
return jniRegisterNativeMethods(env, "com/android/server/SystemServer",
gMethods, NELEM(gMethods));
}

我们能够看到init1被注冊到了android_server_SystemServer_init1 这种方法上了,在android_server_SystemServer_init1 方法中调用了system_init方法。这种方法出如今System_init.cpp中

extern "C" status_t system_init()
{
// 开启传感器服务
SensorService::instantiate();
if (!proc->supportsProcesses()) {
AudioFlinger::instantiate();
// 启动媒体播放服务
MediaPlayerService::instantiate();
CameraService::instantiate();
AudioPolicyService::instantiate();
}
// 启动Android执行时
AndroidRuntime* runtime = AndroidRuntime::getRuntime();
.................
runtime->callStatic("com/android/server/SystemServer", "init2");
.................
return NO_ERROR;

在System_init.cpp类中的system_init()方法中:利用runtime 调用SystemServer的init2方法,init2方法描写叙述例如以下:

public static final void init2() {
// 開始进入Android系统服务
Thread thr = new ServerThread();
thr.setName("android.server.ServerThread");
thr.start();
}

在run方法中:

...........
// 实例化各种系统服务
ContentService.main(context,factoryTest == SystemServer.FACTORY_TEST_LOW_LEVEL);
Slog.i(TAG, "System Content Providers");
ActivityManagerService.installSystemProviders();
Slog.i(TAG, "Battery Service");
battery = new BatteryService(context);
ServiceManager.addService("battery", battery);
Slog.i(TAG, "Lights Service");
lights = new LightsService(context);
............
ServiceManager.addService("vibrator", new VibratorService(context));
............
((ActivityManagerService)ActivityManagerNative.getDefault())
.systemReady(new Runnable() {
public void run() {
........
});

ServerThread线程任务主要是new出系统的服务,然后加入到serviceManager统一管理,最后调用ActivityManagerService的systemReady方法中执行了mMainStack.resumeTopActivityLocked(null);也就是打开了第一个activity。

  final boolean resumeTopActivityLocked(ActivityRecord prev) {
// 寻找没有被finish掉的第一个activity
ActivityRecord next = topRunningActivityLocked(null);
final boolean userLeaving = mUserLeaving;
mUserLeaving = false;
if (next == null) {
if (mMainStack) {
// 没有activity,启动launcher
return mService.startHomeActivityLocked();
}
}
.....................
}

到此,Android世界的第一个activity已经成功启动,它就是Launcher中的主activity。

Android世界第一个activity启动过程的更多相关文章

  1. Android应用程序的Activity启动过程简要介绍和学习计划

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6685853 在Android系统中,Activ ...

  2. 【Android】应用程序Activity启动过程分析

    在Android系统中,有两种操作会引发Activity的启动,一种用户点击应用程序图标时,Launcher会为我们启动应用程序的主Activity:应用程序的默认Activity启动起来后,它又可以 ...

  3. Android深入四大组件(四)Android8.0 根Activity启动过程(前篇)

    前言 在几个月前我写了Android深入四大组件(一)应用程序启动过程(前篇)和Android深入四大组件(一)应用程序启动过程(后篇)这两篇文章,它们都是基于Android 7.0,当我开始阅读An ...

  4. Android 面试必备 - 系统、App、Activity 启动过程“一锅端”

    Android 系统启动过程 从系统层看: linux 系统层 Android系统服务层 Zygote 从开机启动到Home Launcher: 启动bootloader (小程序:初始化硬件) 加载 ...

  5. Android深入四大组件(五)Android8.0 根Activity启动过程(后篇)

    前言 在几个月前我写了Android深入四大组件(一)应用程序启动过程(前篇)和Android深入四大组件(一)应用程序启动过程(后篇)这两篇文章,它们都是基于Android 7.0,当我开始阅读An ...

  6. 根Activity启动过程

    --摘自<Android进阶解密> 根Activity启动过程中会涉及4个进程,分别是Zygote进程.Launcher进程.AMS所在进程(SystemServer进程).应用程序进程, ...

  7. Activity启动过程源代码分析

    事实上写分析源代码文章总会显得非常复杂非常乏味,可是梳理自己看源代码时的一些总结也是一种提高. 这篇博客分析下Activity启动过程源代码,我会尽量说得简单点. 个人的观点是看源代码不能看得太细,否 ...

  8. Android组件体系之Activity启动模式解析

    本文主要分析Activity的启动模式及使用场景. 一.Activity启动模式浅析 1.standard 标准模式,系统默认的启动模式.在启动Activity时,系统总是创建一个新的Activity ...

  9. Android中Activity启动过程探究

    首先追溯到Activity的启动,随便启动一个自己写的demo项目,使用DDMS进行debug标记,然后在Debug中把主线程暂停,可以看到调用栈.如下图所示: 于是我们先看android.app.A ...

随机推荐

  1. WebService常用公共接口

    Web Service 一些对外公开的网络服务接口   商业和贸易: 1.股票行情数据 WEB 服务(支持香港.深圳.上海基金.债券和股票:支持多股票同时查询) Endpoint: http://we ...

  2. CentOS下安装netcat

    CentOS下安装netcat 使用zookeeper过程中,需要监控集群状态.在使用四字命令时(echo conf | nc localhost 2181),报出如下错误:-bash: netcat ...

  3. 采花 flower

    采花 flower 题目描述 萧芸斓是 Z 国的公主,平时的一大爱好是采花. 今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花.花园足够大,容纳 了 n 朵花,花有 c 种颜色(用整数 1- ...

  4. EasyUI序列化提交学习总结

    jquery easyui将form表单元素的值序列化成对象 form表单 <form id="ff"> <input type="text" ...

  5. 微信小程序 body属性的问题

    微信小程序里面没有了body这个节点了,取而代之的是page

  6. ios 瀑布流的那些事情

    转载: 屎壳郎情调-成长日记 首先要知道:瀑布流的核心就是要获取到图片的长宽 网上的很多例子都是加载本地图片的 对于新手而言 改成加载网络图片的确是有点压力的  因为本地的图片 我们是很容易就能获取到 ...

  7. JAVA神操作--使用Arthas线上热更新实战

    热更不规范,同事两行泪 背景 C君是一个javaer,最近在开发用户登出接口的时候,不小心把接口参数拼错了 正确的是: /api/v1/user/logout?referrer=www.javaer. ...

  8. kernel thread vs user thread

    The most important difference is they use different memory, the kernel mode thread can access any ke ...

  9. NOI2001食物链

    描述 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A吃B,B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我们并不知道它到底是哪一种. 有人 ...

  10. C#获取二维数组的行数和列数及其多维。。。

    原文发布时间为:2008-11-26 -- 来源于本人的百度文章 [由搬家工具导入] 有一个二维数组sz[,] 怎样获取sz 的行数和列数呢? sz.GetLength(0) 返回第一维的长度(即行数 ...