Deeply understand the principles of Android plug-in technology

Deeply understand the principles of Android plug-in technology

[[431328]]

Preface

The plug-in technology originally originated from the idea of ​​running apk without installation. This installation-free apk can be understood as a plug-in.

Apps that support plug-in can load and run plug-ins at runtime, so that some uncommon functional modules in the app can be made into plug-ins, which can reduce the size of the installation package on the one hand and realize the dynamic expansion of app functions on the other hand;

Today we will talk about plug-in

1. Introduction to plug-in

1. Introduction to plug-in

In the Android system, applications exist in the form of Apk, and applications need to be installed before they can be used. But in fact, the way to install applications in the Android system is quite simple, which is actually to copy the application Apk to different directories of the system and then unzip it;

Common application installation directories are:

  • /system/app: system applications
  • /system/priv-app: system applications
  • /data/app: User applications

The composition of Apk, a common Apk will contain the following parts:

  • classes.dex: Java code bytecode
  • res: resource directory
  • lib:so directory
  • assets: static assets directory
  • AndroidManifest.xml: manifest file

In fact, after opening the application, the Android system only opens a process, then uses ClassLoader to load classes.dex into the process and executes the corresponding components;

Then you may wonder, since Android itself also uses a reflection-like form to load code execution, why can't we execute the code in an Apk?

This is actually the purpose of plug-in, which allows the code in the Apk (mainly Android components) to run without installation, which can bring a lot of benefits. The most obvious advantage is actually hot update and hot repair through the network;

2. Difficulties of plug-in technology

  • Reflect and execute the code in the plugin Apk (ClassLoader Injection)
  • Allow the system to call components in the plugin Apk (Runtime Container)
  • Correctly identify resources in plugin APK (Resource Injection)

3. Parent delegation mechanism

ClassLoader calls the loadClass method to load the class. The code is as follows:

  1. protected Class<?> loadClass(String className, boolean resolve) throws ClassNotFoundException {
  2. //First search from the loaded classes
  3. Class<?> clazz = findLoadedClass(className);
  4. if (clazz == null ) {
  5. ClassNotFoundException suppressed = null ;
  6. try {
  7. //If it has not been loaded, call the parent loader's loadClass first
  8. clazz = parent.loadClass(className, false );
  9. } catch (ClassNotFoundException e) {
  10. suppressed = e;
  11. }
  12. if (clazz == null ) {
  13. try {
  14. //If the parent loader is not loaded, try to load
  15. clazz = findClass(className);
  16. } catch (ClassNotFoundException e) {
  17. e.addSuppressed(suppressed);
  18. throw e;
  19. }
  20. }
  21. }
  22. return clazz;
  23. }

It can be seen that when ClassLoader loads a class, it first checks whether it has loaded the class. If it has not loaded it, it will first ask the parent loader to load it. If the parent loader cannot load the class, it will call its own findClass method to load it. This mechanism avoids repeated loading of classes to a large extent.

2. Detailed explanation of plug-in

1. ClassLoader Injection

In short, in a plug-in scenario, there will be multiple ClassLoaders in the same process:

  • Host ClassLoader: The host is the installed application, which is automatically created when running
  • Plugin ClassLoader: Created using new DexClassLoader

We call this process ClassLoader injection;

After the injection is completed, all classes from the host are loaded using the host's ClassLoader, and all classes from the plugin Apk are loaded using the plugin ClassLoader;

Due to the parent delegation mechanism of ClassLoader, the system classes are not affected by the class isolation mechanism of ClassLoader, so the host Apk can use the component classes from the plug-in in the host process;

2. Runtime Container

After the ClassLoader is injected, you can use the classes in the plugin Apk in the host process. However, we all know that Android components are started by system calls. Components in an uninstalled Apk are not registered with AMS and PMS. It is like if you directly use startActivity to start a component in a plugin Apk, the system will tell you that it cannot be found.

Our solution is very simple, namely runtime container technology. In simple terms, it is to embed some empty Android components in the host Apk. Taking Activity as an example, I preset a ContainerActivity extends Activity in the host and register it in AndroidManifest.xml;

What it has to do is very simple, that is, to help us as a container for the plug-in Activity. It accepts several parameters from the Intent, which are different information about the plug-in, such as:

  • pluginName;
  • pluginApkPath;
  • pluginActivityName, etc. In fact, the most important ones are pluginApkPath and pluginActivityName. When ContainerActivity starts, we load the plug-in's ClassLoader and Resource, and reflect the Activity class corresponding to pluginActivityName;

When loading is complete, ContainerActivity does two things:

  • Forward all lifecycle callbacks from the system to the plugin Activity
  • Accepts the system call of Activity method and forwards it back to the system

We can complete the first step by overriding the lifecycle method of ContainerActivity, and the second step is to define a PluginActivity. Then, when writing the Activity component in the plugin Apk, we no longer integrate android.app.Activity, but integrate it from our PluginActivity. Later, we can automate this operation by replacing the bytecode. I will explain why later. Let's look at the pseudo code first;

  1. public class ContainerActivity extends Activity {
  2. private PluginActivity pluginActivity;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. String pluginActivityName = getIntent().getString( "pluginActivityName" , "" );
  6. pluginActivity = PluginLoader.loadActivity(pluginActivityName, this);
  7. if (pluginActivity == null ) {
  8. super.onCreate(savedInstanceState);
  9. return ;
  10. }
  11. pluginActivity.onCreate();
  12. }
  13. @Override
  14. protected void onResume() {
  15. if (pluginActivity == null ) {
  16. super.onResume();
  17. return ;
  18. }
  19. pluginActivity.onResume();
  20. }
  21. @Override
  22. protected void onPause() {
  23. if (pluginActivity == null ) {
  24. super.onPause();
  25. return ;
  26. }
  27. pluginActivity.onPause();
  28. }
  29. // ...
  30. }
  31. public class PluginActivity {
  32. private ContainerActivity containerActivity;
  33. public PluginActivity(ContainerActivity containerActivity) {
  34. this.containerActivity = containerActivity;
  35. }
  36. @Override
  37. public <T extends View > T findViewById( int id) {
  38. return containerActivity.findViewById(id);
  39. }
  40. // ...
  41. }
  42. // The actual components written in the plugin `Apk`
  43. public class TestActivity extends PluginActivity {
  44. // ......
  45. }

But the principle is probably as simple as this. Starting the plug-in component requires relying on the container. The container is responsible for loading the plug-in component and completing two-way forwarding, forwarding the life cycle callbacks from the system to the plug-in component, and forwarding the system calls from the plug-in component to the system.

3. Resource Injection

The last thing to talk about is resource injection, which is actually quite important. Android application development actually advocates the concept of separation of logic and resources. All resources (layout, values, etc.) will be packaged into Apk, and then a corresponding R class will be generated, which contains reference ids to all resources;

It is not easy to inject resources. Fortunately, the Android system has left us a way out. The most important ones are these two interfaces:

  • PackageManager#getPackageArchiveInfo: Parse the PackageInfo of an uninstalled Apk according to the Apk path;
  • PackageManager#getResourcesForApplication: Create a Resources instance based on ApplicationInfo;

What we need to do is to use these two methods to create a plugin resource instance when loading the plugin Apk in the above ContainerActivity#onCreate. Specifically, we first use PackageManager#getPackageArchiveInfo to get the PackageInfo of the plugin Apk. After we have the PackageInfo, we can assemble an ApplicationInfo ourselves, and then create a resource instance through PackageManager#getResourcesForApplication. The code is roughly like this:

  1. PackageManager packageManager = getPackageManager();
  2. PackageInfo packageArchiveInfo = packageManager.getPackageArchiveInfo(
  3. pluginApkPath,
  4. PackageManager.GET_ACTIVITIES
  5. | PackageManager.GET_META_DATA
  6. | PackageManager.GET_SERVICES
  7. | PackageManager.GET_PROVIDERS
  8. | PackageManager.GET_SIGNATURES
  9. );
  10. packageArchiveInfo.applicationInfo.sourceDir = pluginApkPath;
  11. packageArchiveInfo.applicationInfo.publicSourceDir = pluginApkPath;
  12. Resources injectResources = null ;
  13. try {
  14. injectResources = packageManager.getResourcesForApplication(packageArchiveInfo.applicationInfo);
  15. } catch (PackageManager.NameNotFoundException e) {
  16. // ...
  17. }

After getting the resource instance, we need to merge the host's resources with the plugin resources and write a new Resources class to complete the automatic proxy in this way:

  1. public class PluginResources extends Resources {
  2. private Resources hostResources;
  3. private Resources injectResources;
  4. public PluginResources(Resources hostResources, Resources injectResources) {
  5. super(injectResources.getAssets(), injectResources.getDisplayMetrics(), injectResources.getConfiguration());
  6. this.hostResources = hostResources;
  7. this.injectResources = injectResources;
  8. }
  9. @Override
  10. public String getString( int id, Object... formatArgs) throws NotFoundException {
  11. try {
  12. return injectResources.getString(id, formatArgs);
  13. } catch (NotFoundException e) {
  14.  
  15. return hostResources.getString(id, formatArgs);
  16. }
  17. }
  18. // ...
  19. }

Then, after ContainerActivity completes loading the plug-in component, we create a Merge resource and then override ContainerActivity#getResources to replace the obtained resources:

  1. public class ContainerActivity extends Activity {
  2. private Resources pluginResources;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. // ...
  6. pluginResources = new PluginResources(super.getResources(), PluginLoader.getResources(pluginApkPath));
  7. // ...
  8. }
  9. @Override
  10. public Resources getResources() {
  11. if (pluginActivity == null ) {
  12. return super.getResources();
  13. }
  14. return pluginResources;
  15. }
  16. }

This completes the injection of resources

4. Resolving resource conflicts

The merged resource processing method will introduce resource conflicts because the resource IDs in different plug-ins may be the same, so the solution is to make different plug-in resources have different resource IDs;

The resource id is represented by an 8-bit hexadecimal number, expressed as 0xPPTTNNNN. The PP segment is used to distinguish the package space. By default, it only distinguishes between application resources and system resources. The TT segment is the resource type. The NNNN segment increases from 0000 in the same APK.

Summarize

There are actually many plug-in frameworks on the market, such as Tecent's Shadow, Didi's VirtualApk, and 360's RePlugin. They each have their own strengths, but they are generally similar;

Their general principles are actually similar. When running, there will be a host Apk running in the process. The dormitory Apk is the actual installed application. The host Apk can load the components and codes in the plug-in Apk to run, and the plug-in Apk can be hot-updated at will;

<<:  Apple iOS/iPadOS 15.1 official version released: support for AirPods 3, sharing and broadcasting, and a series of improvements

>>:  Xiaomi stores are everywhere, but why are there so few Apple stores?

Recommend

Apple Pay: Apple's payment ambitions

This time, money was directly used and Jobs was n...

Why can’t Douyin influencer promotion be charged based on results?

Whether it is performance advertising or brand ad...

Introduction to Glide: Image loading library for Android

As an Android developer, you're probably fami...

21 most commonly used growth techniques by foreign growth hackers

After testing and practicing throughout 2017, for...

Dr. Mo explains the Apple-Google smartphone war: Both giants face challenges

[[124508]] The American technology blog Re/code w...

I have some bad news for you: you may not be suitable for operations!

Continuous self-examination, summarization, and f...

Should you specialize in one area, or move toward full-stack operations?

The concept of full stack originated from full st...

How to improve user stickiness for Internet financial products?

In recent years, Internet finance has become very...