美文网首页
Android四大组件的工作过程,原理解析

Android四大组件的工作过程,原理解析

作者: ChristZc | 来源:发表于2022-07-06 14:51 被阅读0次

    一、Activity启动过程

        首先我们要知道Activity有冷启动和热启动之分,通俗来说冷启动就是应用进程尚未创建,热启动则已经创建完成。

        在点击桌面应用图标时,即将要启动的App将和Launcher、AMS、Zygote这三者多次通信,才会启动一个App,然后再启动Activity。

    追踪源码,我们可以得到以下整体的时序图:

    Activity启动时序图

    简单的梳理下整体流程,我们可以直接得到以下流程:

    Activity启动总体流程图

    Launcher进程通过Binder机制通知AMS

    AMS则判断应用进程是否已经存在(冷启动),不存在则通过Socket通讯通知Zygote进程fork应用进程,已经启动(热启动)则无需再次创建App进程。

    应用进程创建完成之后则会通过Binder机制通知AMS 创建完成。

    AMS通过Binder机制请求ApplicationThread创建并启动根Activity。

    ApplicationThread通过Handle机制通知主线程ActivityThread,最终调用到根Activity的onCreate、onStart、onResume等方法完成创建。

    二、Service的启动过程

        Service的启动还是和Activity有很多相似处的,都需要和AMS打招呼。

        众所周知,Service的启动有两种方式一种StartService,一种bindService,那它们的启动过程是否一样呢?带着疑惑我们往下看。

        首先我们看StartService,直接分析源码,可以得到以下时序图:

    startService时序图1 startService时序图2

        注:1.bringUpServiceLocked方法调用时,其中启动Service时会判断app==null,app的类型是ProcessRecord,用来描述运行的应用程序进程的信息,在其中将Service运行的进程名和uid传递给ActivityManagerService的getProcessRecordLocked方法,从而获取运行Service的应用程序进程信息ProcessRecord,如果用来运行Service的应用程序进程不存在,就会调用ActivityManagerService的startProcessLocked方法来创建对应的应用程序进程;如果用来运行Service的应用程序进程存在,会调用realStartServiceLocked方法。

        2.H是ActivityThread的内部类并继承自Handler,AMS通过IApplicationThread向应用程序进程发送消息,接受消息的操作是在应用程序进程的Binder线程池中进行,因此需要Handler来发送消息切换到主线程。

        总结下来就是

    startService总流程

        下面接着分析下bindService,它的流程会相对startService复杂一丢丢,照例直接分析源码,可以得到以下时序图:

    bindService时序图1 bindService时序图2

        大致总结一下就是应用进程这边调用一系列方法最终与AMS通信bindService,然后AMS中调用activeService类中的方法,然后又通过ApplicationThread与应用进程这边通信handleBindService,然后通知AMS去publishService,调用的activeService的publishServiceLocked方法,其中调用了ServiceDispatcher.InnerConnection的connected方法,再调用ServiceDispatcher的connected方法,ServiceDispatcher是LoadedApk的内部类,并调用Handler类型的对象mActivityThread的post方法,mActivityThread实际指向的是ActivityThread的内部类H,最终通过H类的post方法将RunConnection对象的内容运行在主线程中,RunConnections是LoadedApk的内部类,最后调用mConnection的onServiceConnected,mConnection的类型是ServiceConnection,这样客户端实现ServiceConnection接口类的onServiceConnected方法就会被调用。

    三、BroadcastReceiver的注册、发送和接收

       广播注册分为动态注册和静态注册,首先我们分析动态注册。

        动态注册的过程是从ContextWrapper#registerReceiver()开始的. 和Activity或者Service一样. ContextWrapper并没有做实际的工作, 而是将注册的过程直接交给了ContextImpl来完成。

        ContextImpl#registerReceiver()方法调用了本类的registerReceiverInternal()方法。

        系统首先从mPackageInfo获取到IIntentReceiver对象, 然后再采用跨进程的方式向AMS发送广播注册的请求. 之所以采用IIntentReceiver而不是直接采用BroadcastReceiver, 这是因为上述注册过程中是一个进程间通信的过程. 而BroadcastReceiver作为Android中的一个组件是不能直接跨进程传递的. 所以需要通过IIntentReceiver来实现通信。

        IIntentReceiver作为一个Binder接口, 它的具体实现是LoadedApk.ReceiverDispatcher.InnerReceiver, ReceiverDispatcher的内部同时保存了BroadcastReceiver和InnerReceiver, 这样当接收到广播的时候, ReceiverDispatcher可以很方便的调用BroadcastReceiver#onReceive()方法. 这里和Service很像有同样的类, 并且内部类中同样也是一个Binder接口。

        由于注册广播真正实现过程是在AMS中, 因此跟进AMS中, 首先看registerReceiver()方法, 这里只关心里面的核心部分. 这段代码最终会把远程的InnerReceiver对象以及IntentFilter对象存储起来, 这样整个广播的注册就完成了。

    静态注册的过程就比较简单

    1.AndroidManifest.xml中的receiver解析成ParsedActivity, 存放在PackageImpl的receivers(PackageImpl父类是ParsingPackageImpl而且实现了AndroidPackage接口)

    2.receivers最终会存放在mComponentResolver(PMS)的mReceivers中,mReceivers包含了所有安装应用的receiver,通过ComponentName就可以获得相应的ParsedActivity,如:

    3.mReceivers.mActivities.get(new ComponentName(“com.example.myapplication”, “com.example.myapplication.MyReceive”)),既可以获得MyReceiver的ParsedActivity

    4.receiver中的一个IntentFilter(intent-filter)对应一个ParsedIntentInfo,一个Pair<ParsedActivity, ParsedIntentInfo>

    5.Pair<ParsedActivity, ParsedIntentInfo>被放入mComponentResolver的mFilters

    6.mActions被放入mComponentResolver的mActionToFilter

        部分静态广播注册需要权限,不是注册了就能收到。其实要接收广播还:涉及权限、进程优先级、flag标签等各类无法接收到广播的情况。总而言之静态广播注册是在包安装解析的时候就开始注册,并将广播信息存储在PMS(PackageManagerService)中了。

        广播的发送主要有几种:普通广播、有序广播。

        前面都一样都是通过ContextImpl去与AMS打交道,然后AMS这边方法中会根据intent-filter查找出匹配的广播接收者并经过一系列的条件过滤. 最终会将满足条件的广播接收者添加到BroadcastQueue中, 接着BroadcastQueue就会将广播发送给相应广播接收者。应用接收到广播, 同时onReceive()方法是在广播接收者的主线程中被调用的。

    普通广播发送

    补充小知识

    1:所有静态广播Receiver 都是串行处理,(静态广播接收者属于在发送此广播时属于有序广播范畴)

    2:动态广播 Receiver按照发送此广播时指定的方式 进行 串行或者并行分发

    其中:如果是普通广播:那么通过 并行方式分发

            如果是 有序广播(sendOrderxxxx):那么通过 串行方式分发

    3:Android系统在处理广播时:动态广播接收者 优先于 静态广播接收者 收到

    4:广播分发处理关键类:BroadcastQueue/ BroadcastRecord

    5.本地广播:为了解决全局广播的安全性问题,Android引进了本地广播机制。本地广播发出后只能够在应用程序内部传递,也只有应用程序内部的接收器能接收到本地广播,这样广播的安全性问题就能得到解决了。本地广播和全局广播不同的地方在于本地广播主要使用LocalBroadcastManager对广播的发送、注册、注销进行管理。

    四、ContentProvider的启动过程

        ContentProvider的启动过程和其他3大组件基本类似,都是通过AMS实现进程间的数据共享。

        在activity中如果想通过provider来实现增删查改,首先需要获取contentprovider,大致过程为在context中获取contentResolver,然后通过contentResolver去ActivityManagerService中查询对应的provider,如果没有则进入PackageManagerService中查找:

        1)首先每个context类都会内部包含了一个ContentResolver的子对象ApplicationContentResolver。

        2)通过调用ApplicationContentResolver的主要方法query来获取CP的数据库数据。

        3)调用的过程首先会调用ContentResolver的核心方法acquireProvider()。而acquireProvider()方法是一个抽象方法,其实现是交由子类实现。

        4)通过子类的acquireProvider()方法实现了解到主要的实现是交由ActivityThread类来完成。

        5)ActivityThread类会做出一个判断,如果本地保存一个需要获取的CP对象实例,就会直接返回这个对象实例,如果没有保存,则会访问AMS对象去查找获取一个对象的CP对象实例,当找到这个对象实例后会保存到本地以便日后快速获取。

        6)如果在AMS里面没有找到,就会继续深入到PMS里去从全部的provider中查找。

        7)获取到CP对象实例后会通过层层返回,最后再调用该CP对象的query方法获取相应的数据。

    首先在应用的的manifest中需要进行读写权限申明,这个申明的定义跟之前provider定义中读写所需权限属性值是一样的:

        在activity中获取ContentResolver调用其中的操作方法时,需要传入相对应的参数:

        contentResolver.query(Uri uri, String[] projection,String selection, String[] selectionArgs,String orderBy);

        uri:传入对应uri是为了查找到对应的provider,跟provider在manifest中定义的authorities值是一样

        projection:选择需要返回的对象属性值,有时候不需要将对象的值全部返回。

        selection/selectionArgs:查询条件

        orderBy: 返回的对象排序方式

        类似其他的delete、insert和update操作。最主要的是传入正确的Uri才能找到对应的provider。

        此处加个小知识点:ContentProvider onCreate()优先执行于 Application onCreate()方法,感兴趣的小伙伴可以通过查看源码的方式验证。

    浅谈一个小知识点—AMS的启动:

        AMS,即ActivityManagerService,是安卓java framework的一个服务,运行在system_server进程。

        在系统开机启动之后,system_server会执行三大服务启动

        startBootstrapServices() :启动引导服务,在这里实际上已经new了AMS,在new方法里,已经初始化了AMS的大部分重要的属性。AMS的Looper和各种handler也是在这里准备好的。

        startCoreServices():核心服务。

        在创建完AMS之后,system_server的run方法会执行到startOtherServices(),在启动“其他服务”完毕之后,会调入到AMS的systemReady()方法,在这里会启动launcher。

        可以说,在这个方法执行完毕之后,系统就已经启动完成了。

    相关文章

      网友评论

          本文标题:Android四大组件的工作过程,原理解析

          本文链接:https://www.haomeiwen.com/subject/zqysbrtx.html