美文网首页
Activity getSystemService(String

Activity getSystemService(String

作者: biginsect | 来源:发表于2019-11-24 17:44 被阅读0次

    本文基于Android 9.0(API 28)

    1,先在 Activity 中查找

    在 Activity 的getSystemService() 方法的代码如下:

        @Override
        public Object getSystemService(@ServiceName @NonNull String name) {
            if (getBaseContext() == null) {
                throw new IllegalStateException(
                        "System services not available to Activities before onCreate()");
            }
    
            if (WINDOW_SERVICE.equals(name)) {
                return mWindowManager;
            } else if (SEARCH_SERVICE.equals(name)) {
                ensureSearchManager();
                return mSearchManager;
            }
            return super.getSystemService(name);
        }
    

    1)Activity 检查发现如果需要的是 WindowManagerSearchManager,则返回其属性 mWindowManager 和 mSearchManager。WindowManager 最终也是通过 Context 的getSystemService()获取,而 ensureSearchManager() 在 Activity 中直接创建SearchManager。
    2)否则调用其父类 ContextThemeWrapper对应的方法:

        @Override
        public Object getSystemService(String name) {
            if (LAYOUT_INFLATER_SERVICE.equals(name)) {
                if (mInflater == null) {
                    mInflater = LayoutInflater.from(getBaseContext()).cloneInContext(this);
                }
                return mInflater;
            }
            return getBaseContext().getSystemService(name);
        }
    
    

    可以看到LayoutInflater.from() 传入的是 getBaseContext(),这个方法是ContextThemeWrapper 的父类 ContextWrapper 定义的,返回其属性 mBase(Context类型)。mBase 的赋值有两个地方:
    其一,构造方法

    public ContextWrapper(Context base) {
            mBase = base;
        }
    

    通过全局搜索,没有发现有用的信息。
    其二,

     protected void attachBaseContext(Context base) {
            if (mBase != null) {
                throw new IllegalStateException("Base context already set");
            }
            mBase = base;
        }
    

    这个方法在 Activity 的attach() 方法中调用:

    final void attach(Context context, ActivityThread aThread,
                Instrumentation instr, IBinder token, int ident,
                Application application, Intent intent, ActivityInfo info,
                CharSequence title, Activity parent, String id,
                NonConfigurationInstances lastNonConfigurationInstances,
                Configuration config, String referrer, IVoiceInteractor voiceInteractor,
                Window window, ActivityConfigCallback activityConfigCallback, IBinder assistToken) {
    
            attachBaseContext(context);
            ......
    }
    

    2,先看 ContextImpl 是如何创建的

    我们知道 ActivityThread 的main() 是程序的入口,

    public static void main(String[] args) {
            Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "ActivityThreadMain");
    
            // Install selective syscall interception
            AndroidOs.install();
            ....
    
            Looper.prepareMainLooper();
            ....
            ActivityThread thread = new ActivityThread();
            thread.attach(false, startSeq);
            ...
            Looper.loop();
    }
    

    开始准备 Looper 处理消息,需要注意到这个attach() 方法,传入的 system 参数是false

        @UnsupportedAppUsage
        private void attach(boolean system, long startSeq) {
            sCurrentActivityThread = this;
            mSystemThread = system;
            if (!system) {
                android.ddm.DdmHandleAppName.setAppName("<pre-initialized>",
                                                        UserHandle.myUserId());
                RuntimeInit.setApplicationObject(mAppThread.asBinder());
                final IActivityManager mgr = ActivityManager.getService();
                try {
                    mgr.attachApplication(mAppThread, startSeq);
                } catch (RemoteException ex) {
                    throw ex.rethrowFromSystemServer();
                }
                ...
            } else {//主要是创建application,准备程序所需要的环境等等
               ...
            }
    }
    

    通过 AIDL 获取ActivityManagerService(继承于 IActivityManager.Stub) 代理对象,同时调用attachApplication() 。其内部调用 ApplicationThread(IApplicationThread 类型) 的bindApplication() 方法,最终调用了attachApplicationLocked()。

    @GuardedBy("this")
        private final boolean attachApplicationLocked(IApplicationThread thread,
                int pid, int callingUid, long startSeq) {
                ......
            //  当前application与之前的进程绑定,需要解绑并清空信息
            // If this application record is still attached to a previous
            // process, clean it up now.
            if (app.thread != null) {
                handleAppDiedLocked(app, true, true);
            }
           //准备其他信息,先不关注
            ...
              //创建application并绑定
               checkTime(startTime, "attachApplicationLocked: immediately before bindApplication");
                mStackSupervisor.getActivityMetricsLogger().notifyBindApplication(app);
                if (app.isolatedEntryPoint != null) {
                    // This is an isolated process which should just call an entry point instead of
                    // being bound to an application.
                    thread.runIsolatedEntryPoint(app.isolatedEntryPoint, app.isolatedEntryPointArgs);
                } else if (app.instr != null) {
                    thread.bindApplication(processName, appInfo, providers,
                            app.instr.mClass,
                            profilerInfo, app.instr.mArguments,
                            app.instr.mWatcher,
                            app.instr.mUiAutomationConnection, testMode,
                            mBinderTransactionTrackingEnabled, enableTrackAllocation,
                            isRestrictedBackupMode || !normalMode, app.persistent,
                            new Configuration(getGlobalConfiguration()), app.compat,
                            getCommonServicesLocked(app.isolated),
                            mCoreSettingsObserver.getCoreSettingsLocked(),
                            buildSerial, isAutofillCompatEnabled);
                } else {
                    thread.bindApplication(processName, appInfo, providers, null, profilerInfo,
                            null, null, null, testMode,
                            mBinderTransactionTrackingEnabled, enableTrackAllocation,
                            isRestrictedBackupMode || !normalMode, app.persistent,
                            new Configuration(getGlobalConfiguration()), app.compat,
                            getCommonServicesLocked(app.isolated),
                            mCoreSettingsObserver.getCoreSettingsLocked(),
                            buildSerial, isAutofillCompatEnabled);
                }
                ...
            //启动已经准备好的activity ,开始启动activity
            // See if the top visible activity is waiting to run in this process...
            if (normalMode) {
                try {
                    if (mStackSupervisor.attachApplicationLocked(app)) {
                        didSomething = true;
                    }
                } catch (Exception e) {
                    Slog.wtf(TAG, "Exception thrown launching activities in " + app, e);
                    badApp = true;
                }
            }
            //准备需要的service  
            // Find any services that should be running in this process...
            if (!badApp) {
                try {
                    didSomething |= mServices.attachApplicationLocked(app, processName);
                    checkTime(startTime, "attachApplicationLocked: after mServices.attachApplicationLocked");
                } catch (Exception e) {
                    Slog.wtf(TAG, "Exception thrown starting services in " + app, e);
                    badApp = true;
                }
            }
            ...
            return true;
        }
    
    1. 调用了 IApplicationThread 的 bindApplication() ,实际上是调用了ActivityThread 内部类 ApplicationThread 的 bindApplication()。
    public final void bindApplication(String processName, ApplicationInfo appInfo,
                    List<ProviderInfo> providers, ComponentName instrumentationName,
                    ProfilerInfo profilerInfo, Bundle instrumentationArgs,
                    IInstrumentationWatcher instrumentationWatcher,
                    IUiAutomationConnection instrumentationUiConnection, int debugMode,
                    boolean enableBinderTracking, boolean trackAllocation,
                    boolean isRestrictedBackupMode, boolean persistent, Configuration config,
                    CompatibilityInfo compatInfo, Map services, Bundle coreSettings,
                    String buildSerial, boolean autofillCompatibilityEnabled) {
                ...
                AppBindData data = new AppBindData();
                data.processName = processName;
                data.appInfo = appInfo;
                ...
                sendMessage(H.BIND_APPLICATION, data);
            }
    

    发送了绑定 application 的消息,最终调用了 ActivityThread 的 handleBindApplication()。

    private void handleBindApplication(AppBindData data) {
            // Register the UI Thread as a sensitive thread to the runtime.
            //准备dalvik 需要的数据信息
            ...
           //准备 Instrumentation info
            // Instrumentation info affects the class loader, so load it before
            // setting up the app context.
            final InstrumentationInfo ii;
            if (data.instrumentationName != null) {
                try {
                    ii = new ApplicationPackageManager(null, getPackageManager())
                            .getInstrumentationInfo(data.instrumentationName, 0);
                } catch (PackageManager.NameNotFoundException e) {
                    throw new RuntimeException(
                            "Unable to find instrumentation info for: " + data.instrumentationName);
                }
           .....
             //创建application,内部是使用 ClassLoader 的newInstance() 创建
                Application app;
            try {
                app = data.info.makeApplication(data.restrictedBackupMode, null);
               ...
               }
                try {//回调 application 的 onCreate()。 
                    mInstrumentation.callApplicationOnCreate(app);
                } catch (Exception e) {
                    if (!mInstrumentation.onException(app, e)) {
                        throw new RuntimeException(
                          "Unable to create application " + app.getClass().getName()
                          + ": " + e.toString(), e);
                    }
                }
            } finally {
            ...
            }
            ...
        }
    

    至此,application成功创建,并且比 activity 更早。

    2)发现调用了 ActivityStackSupervisor(用于辅助管理activity栈) 的 attachApplicationLocked(),其实也是调用了 realStartActivityLocked() 方法。

    final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app,
                boolean andResume, boolean checkConfig) throws RemoteException {
            ...
            //创建启动 activity 的事务,需要关注 ClientTransaction
           // Create activity launch transaction.
                    final ClientTransaction clientTransaction = ClientTransaction.obtain(app.thread,
                            r.appToken);
                    clientTransaction.addCallback(LaunchActivityItem.obtain(new Intent(r.intent),
                            System.identityHashCode(r), r.info,
                            // TODO: Have this take the merged configuration instead of separate global
                            // and override configs.
                            mergedConfiguration.getGlobalConfiguration(),
                            mergedConfiguration.getOverrideConfiguration(), r.compat,
                            r.launchedFromPackage, task.voiceInteractor, app.repProcState, r.icicle,
                            r.persistentState, results, newIntents, mService.isNextTransitionForward(),
                            profilerInfo));
    
                    // Set desired final state.
                    final ActivityLifecycleItem lifecycleItem;
                    if (andResume) {
                        lifecycleItem = ResumeActivityItem.obtain(mService.isNextTransitionForward());
                    } else {
                        lifecycleItem = PauseActivityItem.obtain();
                    }
                    clientTransaction.setLifecycleStateRequest(lifecycleItem);
    
                    // Schedule transaction.
                    mService.getLifecycleManager().scheduleTransaction(clientTransaction);
                ...
    }
    

    mService.getLifecycleManager().scheduleTransaction(clientTransaction) 最终会调用到 ClientTransaction的 schedule()

    public void schedule() throws RemoteException {
            mClient.scheduleTransaction(this);
        }
    

    mClient 是 IApplicationThread 类型,其实现类是 ActivityThread 的内部类 ApplicationThread。

           @Override
            public void scheduleTransaction(ClientTransaction transaction) throws RemoteException {
                ActivityThread.this.scheduleTransaction(transaction);
            }
    

    其实是交给 ActivityThread 处理,ActivityThread 没有实现这个方法,其父类 ClientTransactionHandler 中,

    void scheduleTransaction(ClientTransaction transaction) {
            transaction.preExecute(this);
            sendMessage(ActivityThread.H.EXECUTE_TRANSACTION, transaction);
        }
    

    使用了 Handler 机制,对应的 handleMessage()方法中

    case EXECUTE_TRANSACTION:
                        final ClientTransaction transaction = (ClientTransaction) msg.obj;
                        mTransactionExecutor.execute(transaction);
                        if (isSystem()) {
                            // Client transactions inside system process are recycled on the client side
                            // instead of ClientLifecycleManager to avoid being cleared before this
                            // message is handled.
                            transaction.recycle();
                        }
                        // TODO(lifecycler): Recycle locally scheduled transactions.
                        break;
    

    会执行在ActivityStackSupervisor提交的事务,clientTransaction.addCallback(LaunchActivityItem.obtain(new Intent(r.intent),... ...),看到LaunchActivityItem 这个名字就知道这个适用于启动 activity 的,其 execute() 方法

    @Override
        public void execute(ClientTransactionHandler client, IBinder token,
                PendingTransactionActions pendingActions) {
            Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
            ActivityClientRecord r = new ActivityClientRecord(token, mIntent, mIdent, mInfo,
                    mOverrideConfig, mCompatInfo, mReferrer, mVoiceInteractor, mState, mPersistentState,
                    mPendingResults, mPendingNewIntents, mIsForward,
                    mProfilerInfo, client);
            client.handleLaunchActivity(r, pendingActions, null /* customIntent */);
            Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
        }
    

    调用到ClientTransactionHandler的 handleLaunchActivity() 方法,实际上执行的是ActivityThread的 handleLaunchActivity() 方法——》performLaunchActivity() 方法, performLaunchActivity() 会通过Instrumentation加载 activity 及其所需要的资源(Component等)。

    private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
            ActivityInfo aInfo = r.activityInfo;
            if (r.packageInfo == null) {
                r.packageInfo = getPackageInfo(aInfo.applicationInfo, r.compatInfo,
                        Context.CONTEXT_INCLUDE_CODE);
            }
    
            ComponentName component = r.intent.getComponent();
            if (component == null) {
                component = r.intent.resolveActivity(
                    mInitialApplication.getPackageManager());
                r.intent.setComponent(component);
            }
              ...
            ContextImpl appContext = createBaseContextForActivity(r);
            Activity activity = null;
            try {
                java.lang.ClassLoader cl = appContext.getClassLoader();
                activity = mInstrumentation.newActivity(
                        cl, component.getClassName(), r.intent);
                StrictMode.incrementExpectedActivityCount(activity.getClass());
                r.intent.setExtrasClassLoader(cl);
                r.intent.prepareToEnterProcess();
                if (r.state != null) {
                    r.state.setClassLoader(cl);
                }
            } catch (Exception e) {
                }
            }
                    ...
                 activity.attach(appContext, this, getInstrumentation(), r.token,
                            r.ident, app, r.intent, r.activityInfo, title, r.parent,
                            r.embeddedID, r.lastNonConfigurationInstances, config,
                            r.referrer, r.voiceInteractor, window, r.configCallback);
                   ...
                    if (r.isPersistable()) {
                        mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);
                    } else {
                        mInstrumentation.callActivityOnCreate(activity, r.state);
                    }
            ...
            return activity;
        }
    

    到这里,activity 会创建并开始它的生命周期。注意ContextImpl appContext = createBaseContextForActivity(r);,其内部

    private ContextImpl createBaseContextForActivity(ActivityClientRecord r) {
            final int displayId;
            try {
                displayId = ActivityManager.getService().getActivityDisplayId(r.token);
            } catch (RemoteException e) {
                throw e.rethrowFromSystemServer();
            }
    
            ContextImpl appContext = ContextImpl.createActivityContext(
                    this, r.packageInfo, r.activityInfo, r.token, displayId, r.overrideConfig);
    
            final DisplayManagerGlobal dm = DisplayManagerGlobal.getInstance();
            // For debugging purposes, if the activity's package name contains the value of
            // the "debug.use-second-display" system property as a substring, then show
            // its content on a secondary display if there is one.
            String pkgName = SystemProperties.get("debug.second-display.pkg");
            if (pkgName != null && !pkgName.isEmpty()
                    && r.packageInfo.mPackageName.contains(pkgName)) {
                for (int id : dm.getDisplayIds()) {
                    if (id != Display.DEFAULT_DISPLAY) {
                        Display display =
                                dm.getCompatibleDisplay(id, appContext.getResources());
                        appContext = (ContextImpl) appContext.createDisplayContext(display);
                        break;
                    }
                }
            }
            return appContext;
        }
    

    看到返回的是ContextImpl类型的 appContext。因此实际上使用的是ContextImpl对象及其实现的方法。

    3,getSystemService() 最终使用的是什么

    ContextImpl 的对应方法中

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

    发现这些服务都是SystemServiceRegistry管理的,它在静态代码块中注册了对应的服务

    registerService(Context.LAYOUT_INFLATER_SERVICE, LayoutInflater.class,
                    new CachedServiceFetcher<LayoutInflater>() {
                @Override
                public LayoutInflater createService(ContextImpl ctx) {
                    return new PhoneLayoutInflater(ctx.getOuterContext());
                }});
    

    实际上LayoutInflater.from()使用的是PhoneLayoutInflater。相同的还有InputMethodManagerWindowManagerImpl等等。

    4,总结

    至此,我们的源码分析完毕。可以发现Context作为父类,抽象了大部分常用到的方法,如getString(),startActivities(),getResources(),getMainLooper(),getSystemService()等等。其子类ContextWhraper也只是作为一个代理,简单的返回 其属性 mBase (Context类型) 中对应的方法调用结果而已。实际上所有的工作都是由ContextImpl来完成。

    当然在分析的过程中,启动 Activity 时在Manifest 做如下配置:

    <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
    

    就能将对应 Activity 作为程序的主 Activity 的原因还没有分析,需要后面找时间来完成。

    相关文章

      网友评论

          本文标题:Activity getSystemService(String

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