简单跟踪一下getSystemService()

Android  源码分析  2024年3月22日 pm5:55发布8个月前更新 91es.com站长
93 0 0

前言

记录一下getSystemService()获取的源码流程,加深一下印象。

正文

这里以获取AudioManager为例

AudioManager audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);

getSystemService()定义在Context.java中

public static final String AUDIO_SERVICE = "audio";

Context.java

frameworks\base\core\java\android\content\Context.java
public abstract @Nullable Object getSystemService(@ServiceName @NonNull String name);

抽象方法。

Context的实现类是ContextImpl,具体看ContextImpl.java

ContextImpl.java

\frameworks\base\core\java\android\app\ContextImpl.java
@Override
public Object getSystemService(String name) {
    if (vmIncorrectContextUseEnabled()) {
        //略
    }
    //重点关注后面的
    return SystemServiceRegistry.getSystemService(this, name);
}

SystemServiceRegistry.java

\frameworks\base\core\java\android\app\SystemServiceRegistry.java

我们一开始传入的name是Context.AUDIO_SERVICE,也就是audio。

public static Object getSystemService(ContextImpl ctx, String name) {
    //不为null
    if (name == null) {
        return null;
    }
    //SYSTEM_SERVICE_FETCHERS是map,存储了服务相关信息
    final ServiceFetcher<?> fetcher = SYSTEM_SERVICE_FETCHERS.get(name);
    if (fetcher == null) {
        if (sEnableServiceNotFoundWtf) {
            Slog.wtf(TAG, "Unknown manager requested: " + name);
        }
        return null;
    }
    //获取服务,这里需要重点关注fetcher
    final Object ret = fetcher.getService(ctx);
    if (sEnableServiceNotFoundWtf && ret == null) {
        // Some services do return null in certain situations, so don't do WTF for them.
        switch (name) {
            case Context.CONTENT_CAPTURE_MANAGER_SERVICE:
            case Context.APP_PREDICTION_SERVICE:
            case Context.INCREMENTAL_SERVICE:
            case Context.ETHERNET_SERVICE:
                return null;
        }
        return null;
    }
    return ret;
}

上面重点关注的是SYSTEM_SERVICE_FETCHERS和ServiceFetcher。

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

具体看哪里添加的

registerService
//私有的静态变量,类内部使用
private static <T> void registerService(@NonNull String serviceName,
        @NonNull Class<T> serviceClass, @NonNull ServiceFetcher<T> serviceFetcher) {
    SYSTEM_SERVICE_NAMES.put(serviceClass, serviceName);
    //这里添加的
    SYSTEM_SERVICE_FETCHERS.put(serviceName, serviceFetcher);
    SYSTEM_SERVICE_CLASS_NAMES.put(serviceName, serviceClass.getSimpleName());
}

这里有多个map存储服务相关信息,我们这里值关注SYSTEM_SERVICE_FETCHERS。

第一个是服务名,第二个是serviceFetcher。

在SystemServiceRegistry中有一个静态代码块,这里注册了

Context.ACTIVITY_SERVICE
Context.AUDIO_SERVICE
Context.ACCOUNT_SERVICE
Context.ACCESSIBILITY_SERVICE
Context.HDMI_CONTROL_SERVICE
等

这里只关注AUDIO_SERVICE的注册。

static{
	//略
	registerService(Context.AUDIO_SERVICE, AudioManager.class,
        new CachedServiceFetcher<AudioManager>() {
    @Override
    public AudioManager createService(ContextImpl ctx) {
        return new AudioManager(ctx);
    }});
	//略
}

第三个是匿名创建CachedServiceFetcher

CachedServiceFetcher
static abstract class CachedServiceFetcher<T> implements ServiceFetcher<T> {
    private final int mCacheIndex;

    CachedServiceFetcher() {
        mCacheIndex = sServiceCacheSize++;
    }

    @Override
    @SuppressWarnings("unchecked")
    public final T getService(ContextImpl ctx) {
        final Object[] cache = ctx.mServiceCache;
        final int[] gates = ctx.mServiceInitializationStateArray;
        boolean interrupted = false;
        T ret = null;
		//循环,知道获取成功就退出
        for (;;) {
            boolean doInitialize = false;
            synchronized (cache) {
				//如果存在缓存,就返回
                T service = (T) cache[mCacheIndex];
                if (service != null) {
                    ret = service;
                    break; // exit the for (;;)
                }
                if (gates[mCacheIndex] == ContextImpl.STATE_READY
                        || gates[mCacheIndex] == ContextImpl.STATE_NOT_FOUND) {
                    gates[mCacheIndex] = ContextImpl.STATE_UNINITIALIZED;
                }
                if (gates[mCacheIndex] == ContextImpl.STATE_UNINITIALIZED) {
                    doInitialize = true;
                    gates[mCacheIndex] = ContextImpl.STATE_INITIALIZING;
                }
            }
            if (doInitialize) {
                T service = null;
                @ServiceInitializationState int newState = ContextImpl.STATE_NOT_FOUND;
                try {
                    service = createService(ctx);
                    newState = ContextImpl.STATE_READY;

                } catch (ServiceNotFoundException e) {
                    onServiceNotFound(e);

                } finally {
                    synchronized (cache) {
                        cache[mCacheIndex] = service;
                        gates[mCacheIndex] = newState;
                        cache.notifyAll();
                    }
                }
                ret = service;
                break;
            }
            synchronized (cache) {
                while (gates[mCacheIndex] < ContextImpl.STATE_READY) {
                    try {
                        interrupted |= Thread.interrupted();
                        cache.wait();
                    } catch (InterruptedException e) {
                        interrupted = true;
                    }
                }
            }
        }
        if (interrupted) {
            Thread.currentThread().interrupt();
        }
        return ret;
    }

    public abstract T createService(ContextImpl ctx) throws ServiceNotFoundException;
}

上面代码很多,但最重要的是需要重写的createService()。

因此回到上面第三个参数创建匿名CachedServiceFetcher对象代码处。

new CachedServiceFetcher<AudioManager>() {
    @Override
    public AudioManager createService(ContextImpl ctx) {
        return new AudioManager(ctx);
}}

这里是直接返回了新创建的AudioManager对象。

至此,上面

AudioManager audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);

就类似等价于

AudioManager audioManager =  new AudioManager(ctx);

参考文章

 历史上的今天

  1. 2023: Android加载动画常用做法简介(0条评论)
版权声明 1、 本站名称: 91易搜
2、 本站网址: 91es.com
3、 本站内容: 部分来源于网络,仅供学习和参考,若侵权请留言
3、 本站申明: 个人流水账日记,内容并不保证有效

暂无评论

暂无评论...

随机推荐

[摘]Android如何设置TextView的行间距、行高。

在Android系统中TextView默认行间距比较窄,不美观。我们可以设置每行的行间距,可以通过属性android:lineSpacingExtra或android:lineSpacingMultiplier来做。修改行间距、行高你可以使用如下TextView属性1、android:lin...

蒋勋:人生,即是修行

选择修行的空间人类的空间感是非常奇怪的东西。过去的人从西门町走路到北门,再从北门走到南门,就是台北市的范围了。可是今日你问任何一个小学生,他都会觉得很近,他坐上公交车、地铁就可以到更远的地方。人类在整个工业革命之后,空间不断在扩大。我自己读完大学,要留学的时候,坐飞机还是一件大事,做这件事情之...

李娟:通往滴水泉的路

最早的时候,通往滴水泉的路只有“乌斯曼小道”。乌斯曼是一百年前那个鼎鼎有名的阿尔泰土匪头子,被称为“哈萨克王”。而更早的一些时候,在这茫茫戈壁,所有的路都只沿着其边缘远远绕过。那些路断断续续地,虚弱地进行在群山褶皱之中,遥遥连接着阿尔泰的绿洲和南方的草原雪山。没有人能从这片荒原的腹心通过。没有水,...

bootprof文件分析系统开机时间

前言 开机时间较长,无法满足客户及内部测试指标要求。这里以bootprof文件为例,简单分析各个阶段耗时情况。正文这里以ATC平台为例。bootprof文件在root下,可以使用如下命令获取bootprofadb pull proc/bootprofbootprof内容简介---...

Android跳转WIFI界面的几种方式

前言简单记录一下Android跳转WiFi设置界面的启动方式。PS: 本文摘抄的,方便自己查阅。正文下面几种方式在Android P上测试OK。第一种try { Intent intent = new Intent(); intent.setAction("androi...

许立志:我谈到血

我谈到血,也是出于无奈我也想谈谈风花雪月谈谈前朝的历史,酒中的诗词可现实让我只能谈到血血源自火柴盒般的出租屋这里狭窄,逼仄,终年不见天日挤压着打工仔打工妹失足妇女异地丈夫卖麻辣烫的四川小伙摆地摊的河南老人以及白天为生活而奔波黑夜里睁着眼睛写诗的我我向你们谈到这些人,谈到我们...