Android Context getSystemService分析

我们知道一个应用的Context个数是Activity个数+Service个数+1

当我们希望获取到系统服务时,可以调用Context的getSystemService方法,如获取到ActivityManager:

ActivityManager manager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);

那么getSystemService又是怎么工作的呢?

Activity是一个Context,他调用getSystemService时,会调用到Context的包装类ContextWrapper的getSystemService方法,如下:

@Override
public Object getSystemService(String name) {
    return mBase.getSystemService(name);
}

mBase是Context的实现类ContextImpl,很明显ContextWrapper又会委托ContextImpl去实现具体逻辑。

我们跟进ContextImpl中的getSystemService方法,如下:

@Override
public Object getSystemService(String name) {
    return SystemServiceRegistry.getSystemService(this, name);
}

其内部是调用了SystemServiceRegistry的getSystemService方法,这时获取系统服务的过程就转移到了SystemServiceRegistry了。

我们看到SystemServiceRegistry的getSystemService方法即可,如下:

/**
 * Gets a system service from a given context.
 */
public static Object getSystemService(ContextImpl ctx, String name) {
    ServiceFetcher<?> fetcher = SYSTEM_SERVICE_FETCHERS.get(name);
    return fetcher != null ? fetcher.getService(ctx) : null;
}

这好像还看不出太多的信息。SYSTEM_SERVICE_FETCHERS.get(name)是什么?

SYSTEM_SERVICE_FETCHERS是一个存储ServiceFetcher的HashMap

private static final HashMap<String, ServiceFetcher<?>> SYSTEM_SERVICE_FETCHERS =
        new HashMap<String, ServiceFetcher<?>>();

我们先不管SYSTEM_SERVICE_FETCHERS.get(name)返回值是什么,

先发现ServiceFetcher是什么时候存储到SYSTEM_SERVICE_FETCHERS里的。

SystemServiceRegistry中有一段静态代码块长这样:

static {

     //代码省略

registerService(Context.ACTIVITY_SERVICE, ActivityManager.class,
            new CachedServiceFetcher<ActivityManager>() {
        @Override
        public ActivityManager createService(ContextImpl ctx) {
            return new ActivityManager(ctx.getOuterContext(), ctx.mMainThread.getHandler());
        }});

    registerService(Context.ALARM_SERVICE, AlarmManager.class,
            new CachedServiceFetcher<AlarmManager>() {
        @Override
        public AlarmManager createService(ContextImpl ctx) {
            IBinder b = ServiceManager.getService(Context.ALARM_SERVICE);
            IAlarmManager service = IAlarmManager.Stub.asInterface(b);
            return new AlarmManager(service, ctx);
        }});

     //代码省略
}

静态代码块只在虚拟机第一次加载类的时候执行。

静态代码块中调用registerService方法来注册系统服务的,而registerService方法内部是将key和value存进去SYSTEM_SERVICE_FETCHERS,找到了,如下:

/**
 * Statically registers a system service with the context.
 * This method must be called during static initialization only.
 */
private static <T> void registerService(String serviceName, Class<T> serviceClass,
        ServiceFetcher<T> serviceFetcher) {
    SYSTEM_SERVICE_NAMES.put(serviceClass, serviceName);
    SYSTEM_SERVICE_FETCHERS.put(serviceName, serviceFetcher);
}

可以看到存进了serviceFetcher,而serviceName就是我们传进的字符串,比如

Context.ACTIVITY_SERVICE

我们分析ActivityManager即可,在注册ActivityManager时传进registerService方法的是CachedServiceFetcher,那么我们跟进CachedServiceFetcher看看。

/**
 * Override this class when the system service constructor needs a
 * ContextImpl and should be cached and retained by that context.
 */
static abstract class CachedServiceFetcher<T> implements ServiceFetcher<T> {
    private final int mCacheIndex;

    public CachedServiceFetcher() {
        mCacheIndex = sServiceCacheSize++;
    }

    @Override
    @SuppressWarnings("unchecked")
    public final T getService(ContextImpl ctx) {
        final Object[] cache = ctx.mServiceCache;
        synchronized (cache) {
            // Fetch or create the service.
            Object service = cache[mCacheIndex];
            if (service == null) {
                service = createService(ctx);
                cache[mCacheIndex] = service;
            }
            return (T)service;
        }
    }

    public abstract T createService(ContextImpl ctx);
}

CachedServiceFetcher实现了ServiceFetcher接口,这个getService方法是在一开始的getSystemService方法中触发,如下

return fetcher != null ? fetcher.getService(ctx) : null;

喔,原来是通过ServiceFetcher来获取系统服务的。

到这里我们分析CachedServiceFetcher可以知道,当第一次获取服务的时候,会调用createService去创建实例,然后将服务对象添加进缓存。下次再取时就直接从缓存中获取了。其实这是一种使用容器来实现的单例模式。

最后总结一下:当我们调用Context的getSystemService方法获取服务时,会转到SystemServiceRegistry类的getSystemService方法,在这个getSystemService方法中,根据key,也就是我们传入的字符串,比如Context.ACTIVITY_SERVICE。根据传入的key从SYSTEM_SERVICE_FETCHERS这个HashMap中获取到ServiceFetcher实例,再调用ServiceFetcher实例的getService方法获取到系统服务对象。如果是第一次获取系统服务的话,会调用ServiceFetcher的createService方法去创建系统服务实例并将其加入缓存列表,之后再次获取就是从缓存中取出了。

而静态代码块又是怎么回事?其实就是为每一个系统服务执行了以下的代码嘛

SYSTEM_SERVICE_NAMES.put(serviceClass, serviceName);
SYSTEM_SERVICE_FETCHERS.put(serviceName, serviceFetcher);

SystemServiceRegistry中的静态代码块只会执行一次,告诉系统,我是有这些服务的,将一些key和value添加进HashMap。就是一些初始化的工作。而并没有真正去创建系统服务对象,因为createService还没有被调用啊

好了,以上就是Context的getSystemService方法的工作过程分析。从中我学到了原来还有这种使用容器的单例模式。

转载请注明出处:http://blog.csdn.net/xyh269

时间: 2024-12-17 05:38:38

Android Context getSystemService分析的相关文章

Android操作系统服务(Context.getSystemService() )

getSystemService是Android很重要的一个API,它是Activity的一个方法,根据传入的NAME来取得对应的Object,然后转换成相应的服务对象.下面介绍系统相应的服务: 传入的Name 返回的对象 说明 WINDOW_SERVICE  WindowManager 管理打开的窗口程序 LAYOUT_INFLATER_SERVICE LayoutInflater  取得xml里定义的view ACTIVITY_SERVICE ActivityManager  管理应用程序的

Android图片缓存分析(一)

Android中写应用时,经常会遇到加载图片的事,由于很多图片是网络上下载获取的,当我们进页面时,便会去网络下载图片,一两次可能没啥问题,但如果同一张图片每次都去网络拉取,不仅速度慢,更影响用户体验,同时会浪费用户的流量. 基于此,很多人便想到了图片缓存的方法. 现在比较普遍的图片缓存主要有以下几个步骤: 一.从缓存中获取图片 二.如果缓存中未获取图片,则从存储卡中获取 三.如果存储卡中未获取图片,则从网络中获取 一.从缓存中获取图片 我们知道,Android中分配给每个应用的内存空间是有限的,

【转】android SystemUI 流程分析

android4 SystemUI 流程分析 什么是SystemUI? 对于Phone来说SystemUI指的是:StatusBar(状态栏).NavigationBar(导航栏).而对于Tablet或者是TV来说SystemUI指的是:CombinedBar(包括了StatusBar和NavigationBar). 启动后Phone界面上的信号,蓝牙标志,Wifi标志等等这些状态显示标志都会在StatusBar上显示.当我们的设备开机后,首先需要给用户呈现的就是各种界面同时也包括了我们的Sys

Android LayoutInflater原理分析,带你一步步深入了解View(一)

转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/12921889 有段时间没写博客了,感觉都有些生疏了呢.最近繁忙的工作终于告一段落,又有时间写文章了,接下来还会继续坚持每一周篇的节奏. 有 不少朋友跟我反应,都希望我可以写一篇关于View的文章,讲一讲View的工作原理以及自定义View的方法.没错,承诺过的文章我是一定要兑现的,而 且在View这个话题上我还准备多写几篇,尽量能将这个知识点讲得透彻一些.那么今天就从Layout

Android内存管理分析

大部分因为工作任务繁重,一般我们很少关心内存的事,只知道先把任务完成.只有真正到了发现UI卡顿 或者APP实在跑不下去了(一点一卡),才会考虑到内存优化.或者你所在的大公司比较关心手机运行流利程度,也需要对内存进行管理. 1.内存管理的基础知识 因为安卓的顶层也是 Java来实现的,作为客户顿的程序员应该懂得如何去管理内存. 又因为Java不像C语言可以执行free去主动释放内存,而是提供了一套Java的垃圾处理器.但是该处理器并不能时刻盯着内存,在内存不需要的时候直接清理(程序员比较方便,但是

Android 下载模块分析(DownloadManager和DownloadProvider)

Android下载模块主要有2个部分组成:DownloadManager和DownloadProvider:其中DownloadManager提供接口供调用,具体的实现是 DownloadProvider,包括相关数据信息的保存及文件下载. DownloadManager是系统开放给第三方应用使用的类,包含两个静态内部类DownloadManager.Query和DownloadManager.Request. DownloadManager.Request用来请求一个下载 DownloadMa

*Android LayoutInflater原理分析

相信接触Android久一点的朋友对于LayoutInflater一定不会陌生,都会知道它主要是用于加载布局的.而刚接触Android的朋友可能对LayoutInflater不怎么熟悉,因为加载布局的任务通常都是在Activity中调用setContentView()方法来完成的.其实setContentView()方法的内部也是使用LayoutInflater来加载布局的,只不过这部分源码是internal的,不太容易查看到.那么今天我们就来把LayoutInflater的工作流程仔细地剖析一

Android Context 是什么?

andorid 开发(42)  版权声明:本文为博主原创文章,未经博主允许不得转载. [转载请注明出处:http://blog.csdn.net/feiduclear_up CSDN 废墟的树] PS:修该了一些有误区的地方. 引言 Context对于Android开发人员来说并不陌生,项目中我们会经常使用Context来获取APP资源,创建UI,获取系统Service服务,启动Activity,绑定Service,发送广播,获取APP信息等等.那么Context到底是什么?Context又是怎

CM android的CMUpdater分析(二)

至于为何要在这里讲解android系统源码中的系统更新,我已经在上一篇< CM android的CMUpdater分析(一)>中介绍了.在上一篇中,主要讲解了在eclipse中如何搭建系统应用的开发环境,现在我们就使用eclipse来分析CMUpdater源码.该系统更新是CM修改原生android的基础上实现的.通过分析android系统的应用源码,可以学到一些很好的思想和编程方法学.好了,废话少说,现在就开始我们的学习之旅. 首先,在开始介绍之前,我先把之前根据CMUpdater源码分析来