Android dynamic proxy and using dynamic proxy to implement ServiceHook

Android dynamic proxy and using dynamic proxy to implement ServiceHook

Java's dynamic proxy

The first thing we want to introduce is Java dynamic proxy. Java dynamic proxy involves two classes: InvocationHandler interface and Proxy class. We will focus on these two classes below, and analyze the correct usage with examples. Before that, let's briefly introduce the generation and loading process of class files in Java. After the Java compiler compiles the Java file, it will generate a .class file on the disk. This .class file is a binary file, and the content is machine code that only the JVM virtual machine can recognize. The JVM virtual machine reads the bytecode file, extracts the binary data, loads it into memory, parses the information in the .class file, and uses the corresponding ClassLoader class loader to generate the corresponding Class object:

The .class bytecode file is generated according to the bytecode organization rules specified in the JVM virtual machine specification. For a detailed introduction to the .class file format, please refer to the blog In-depth Understanding of Java Class File Format and Java Virtual Machine Specification.

From the above, we know that JVM loads classes through binary information of bytecode. If we follow the format and structure of .class files organized by the Java compiler system in the runtime system, generate corresponding binary data, and then convert this binary data into the corresponding class, we can dynamically generate a class we want during operation:

There are many frameworks in Java that can dynamically generate corresponding .class binary bytecodes according to JVM specifications at runtime, such as ASM and Javassist, etc. I will not introduce them in detail here. If you are interested, you can refer to relevant information. Here we take the dynamic proxy mode as an example to introduce the two very important classes we will use. Regarding the dynamic proxy mode, I have already introduced it in Java/Android Design Pattern Learning Notes (9) - Proxy Mode, but I did not analyze the InvocationHandler interface and Proxy class in detail at that time. Here I will introduce it in detail. In the blog on proxy mode, we mentioned that the proxy mode is divided into dynamic proxy and static proxy:

The above is the class diagram of the static proxy mode. When this proxy relationship is specified in the code stage, the ProxySubject class generates a .class bytecode file through the compiler. Before the system runs, this .class file already exists. The structure of the dynamic proxy mode is slightly different from the structure of the static proxy mode above. It introduces an InvocationHandler interface and a Proxy class. In the static proxy mode, the methods in the proxy class ProxySubject are all specifically called to the corresponding methods in a specific RealSubject. What ProxySubject does is nothing more than calling the corresponding method that triggers the RealSubject; the basic mode of dynamic proxy work is to hand over the implementation of its own method functions to the InvocationHandler role. The Proxy role will hand over the calls to each method in the Proxy role to the InvocationHandler for processing, and the InvocationHandler calls the method of RealSubject, as shown in the following figure:

InvocationHandler interface and Proxy class

Let's analyze the steps of generating ProxySubject in dynamic proxy mode:

  1. Get a list of all interfaces on RealSubject;
  2. Determine the class name of the proxy class to be generated. The default name generated by the system is: com.sun.proxy.$ProxyXXXX;
  3. According to the interface information that needs to be implemented, dynamically create the bytecode of the ProxySubject class in the code;
  4. Convert the corresponding bytecode into the corresponding Class object;
  5. Create an instance object h of InvocationHandler to handle all method calls of the Proxy role;
  6. Instantiate a Proxy role object with the created h object as a parameter.

The specific code is:

Subject.java

  1. public interface Subject {
  2. String operation();
  3. }

RealSubject.java

  1. public class RealSubject implements Subject{
  2. @Override
  3. public String operation() {
  4. return   "operation by subject" ;
  5. }
  6. }

ProxySubject.java

  1. public class ProxySubject implements InvocationHandler{
  2. protected Subject subject;
  3. public ProxySubject(Subject subject) {
  4. this.subject = subject;
  5. }
  6.  
  7. @Override
  8. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  9. //do something before
  10. return method.invoke(subject, args);
  11. }
  12. }

Test code

  1. Subject subject = new RealSubject();
  2. ProxySubject proxy = new ProxySubject(subject);
  3. Subject sub = (Subject) Proxy.newProxyInstance(subject.getClass().getClassLoader(),
  4. subject.getClass().getInterfaces(), proxy);
  5. sub.operation();

The above is the simplest implementation code of the dynamic proxy mode. JDK supports dynamic proxy by using the java.lang.reflect.Proxy package. Let's take a look at the description of this class:

  1. Proxy provides static methods for creating dynamic proxy classes and instances, and it is also the
  2. superclass of   all   dynamic proxy classes created by those methods.

Generally, we use the following

  1. public   static Object newProxyInstance(ClassLoader loader, Class<?>[]interfaces,InvocationHandler h) throws IllegalArgumentException {
  2. // Check if h is not empty, otherwise throw an exception
  3. if (h == null ) {
  4. throw new NullPointerException();
  5. }
  6.  
  7. // Get the proxy class type object associated with the specified class loader and a set of interfaces
  8. Class cl = getProxyClass(loader, interfaces);
  9.  
  10. // Get the constructor object through reflection and generate a proxy class instance
  11. try {
  12. Constructor cons = cl.getConstructor(constructorParams);
  13. return (Object) cons.newInstance(new Object[] { h });
  14. } catch (NoSuchMethodException e) { throw new InternalError(e.toString());
  15. } catch (IllegalAccessException e) { throw new InternalError(e.toString());
  16. } catch (InstantiationException e) { throw new InternalError(e.toString());
  17. } catch (InvocationTargetException e) { throw new InternalError(e.toString());
  18. }
  19. }

The getProxyClass method of the Proxy class calls the generatorProxyClass method of the ProxyGenerator to generate a dynamic class:

  1. public   static byte[] generateProxyClass(final String name , Class[] interfaces)

We will introduce this method below, so we will skip it here. After generating the bytecode of this dynamic class, we generate the object of this dynamic class through reflection. After generating a dynamic proxy object sub through this static function of the Proxy class, we call each method of the sub proxy object. Inside the code, we directly call the invoke method of InvocationHandler. The invoke method distinguishes what method it is based on the method parameter passed to it by the proxy class. Let's take a look at the introduction of the InvocationHandler class:

  1. InvocationHandler is the interface implemented by the invocation handler of a proxy instance.
  2.  
  3. Each proxy instance has an associated invocation handler. When a method is invoked on a proxy
  4. instance, the method invocation is encoded and dispatched to the invoke method of its invocation handler.
Public methods
abstract Object invoke(Object proxy, Method method, Object[] args)Processes a method invocation on a proxy instance and returns the result.

Method parameters and return:

Parameters
proxy Object: the proxy instance that the method was invoked on
method Method: the Method instance corresponding to the interface method invoked on the proxy instance. The declaring class of the Method object will be the interface that the method was declared in, which may be a superinterface of the proxy interface that the proxy class inherits the method through.
args Object: an array of objects containing the values ​​of the arguments passed in the method invocation on the proxy instance, or null if interface method takes no arguments. Arguments of primitive types are wrapped in instances of the appropriate primitive wrapper class, such as java.lang.Integer or java.lang.Boolean.
Returns
Object the value to return from the method invocation on the proxy instance. If the declared return type of the interface method is a primitive type, then the value returned by this method must be an instance of the corresponding primitive wrapper class; otherwise, it must be a type assignable to the declared return type. method's declared return type as described above, a ClassCastException will be thrown by the method invocation on the proxy instance.

One point mentioned above that requires special attention is that if the return value of the method defined in the Subject class is one of the eight basic data types, then the corresponding basic type wrapper class must be returned in the ProxySubject class, that is, int corresponds to Integer, etc. It should also be noted that if null is returned at this time, a NullPointerException will be thrown. In other cases, the return value object must be consistent with the return value of the method defined in the Subject class, otherwise a ClassCastException will be thrown.

Generate source code analysis

So what does the class dynamically generated by the newProxyInstance method of the Proxy class look like? As we mentioned above, JDK provides us with a method ProxyGenerator.generateProxyClass(String proxyName,class[] interfaces) to generate the bytecode of the dynamic proxy class. This class is located in the sun.misc package and belongs to a special jar package. So the problem comes again. The android project created by Android studio cannot find the ProxyGenerator class. This class is in the jre directory. Even if I copy the .jar package related to this class into the project and reference it in gradle, although I can find this class in the end, there will be very strange problems during compilation. So, there is no way. Android studio cannot create ordinary java projects. I can only install an intellij idea myself or ask for help from relevant colleagues. After creating a java project, use the following code to export the generated class to the specified path:

  1. public   static void generateClassFile(Class clazz,String proxyName)
  2. {
  3. //Generate bytecode based on the class information and the provided proxy class name
  4. byte[] classFile = ProxyGenerator.generateProxyClass(proxyName, clazz.getInterfaces());
  5. String paths = "D:\\" ; // The path here is hard-coded as drive D, and can be modified according to actual needs
  6. System. out .println(paths);
  7. FileOutputStream out = null ;
  8.  
  9. try {
  10. //Save to hard disk
  11. out = new FileOutputStream(paths+proxyName+ ".class" );
  12. out .write(classFile);
  13. out .flush();
  14. } catch (Exception e) {
  15. e.printStackTrace();
  16. finally
  17. try {
  18. out . close ();
  19. } catch (IOException e) {
  20. e.printStackTrace();
  21. }
  22. }
  23. }

The way to call the code is:

  1. generateClassFile(ProxySubject.class, "ProxySubject" );

Finally, a ProxySubject.class file will be generated in the root directory of the D drive (if the path is not modified). You can open the file using jd-gui:

  1. import java.lang.reflect.InvocationHandler;
  2. import java.lang.reflect.Method;
  3. import java.lang.reflect.Proxy;
  4. import java.lang.reflect.UndeclaredThrowableException;
  5.  
  6. public final class ProxySubject
  7. extends Proxy
  8. implements Subject
  9. {
  10. private static Method m1;
  11. private static Method m3;
  12. private static Method m2;
  13. private static Method m0;
  14.  
  15. public ProxySubject(InvocationHandler paramInvocationHandler)
  16. {
  17. super(paramInvocationHandler);
  18. }
  19.  
  20. public final boolean equals(Object paramObject)
  21. {
  22. try
  23. {
  24. return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
  25. }
  26. catch (Error|RuntimeException localError)
  27. {
  28. throw localError;
  29. }
  30. catch (Throwable localThrowable)
  31. {
  32. throw new UndeclaredThrowableException(localThrowable);
  33. }
  34. }
  35.  
  36. public final String operation()
  37. {
  38. try
  39. {
  40. return (String)this.h.invoke(this, m3, null );
  41. }
  42. catch (Error|RuntimeException localError)
  43. {
  44. throw localError;
  45. }
  46. catch (Throwable localThrowable)
  47. {
  48. throw new UndeclaredThrowableException(localThrowable);
  49. }
  50. }
  51.  
  52. public final String toString()
  53. {
  54. try
  55. {
  56. return (String)this.h.invoke(this, m2, null );
  57. }
  58. catch (Error|RuntimeException localError)
  59. {
  60. throw localError;
  61. }
  62. catch (Throwable localThrowable)
  63. {
  64. throw new UndeclaredThrowableException(localThrowable);
  65. }
  66. }
  67.  
  68. public final int hashCode()
  69. {
  70. try
  71. {
  72. return (( Integer )this.h.invoke(this, m0, null )).intValue();
  73. }
  74. catch (Error|RuntimeException localError)
  75. {
  76. throw localError;
  77. }
  78. catch (Throwable localThrowable)
  79. {
  80. throw new UndeclaredThrowableException(localThrowable);
  81. }
  82. }
  83.  
  84. static  
  85. {
  86. try
  87. {
  88. m1 = Class.forName( "java.lang.Object" ).getMethod( "equals" , new Class[] { Class.forName( "java.lang.Object" ) });
  89. m3 = Class.forName( "Subject" ).getMethod( "operation" , new Class[0]);
  90. m2 = Class.forName( "java.lang.Object" ).getMethod( "toString" , new Class[0]);
  91. m0 = Class.forName( "java.lang.Object" ).getMethod( "hashCode" , new Class[0]);
  92. return ;
  93. }
  94. catch (NoSuchMethodException localNoSuchMethodException)
  95. {
  96. throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
  97. }
  98. catch (ClassNotFoundException localClassNotFoundException)
  99. {
  100. throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
  101. }
  102. }
  103. }

It can be observed that this generated class inherits from java.lang.reflect.Proxy and implements the Subject interface. Let's take a look at the code for generating dynamic classes:

  1. Subject sub = (Subject) Proxy.newProxyInstance(subject.getClass().getClassLoader(),
  2. subject.getClass().getInterfaces(), proxy);

It can be seen that this dynamically generated class will implement all interfaces in subject.getClass().getInterfaces(), and another point is that all methods in the class are final, and the class is also final, so the class cannot be inherited. Finally, all methods will call the invoke() method of the InvocationHandler object h, which is why the invoke() method of the ProxySubject class is finally called. Draw their simple class diagram:

From this class diagram, we can clearly see that the dynamically generated class ProxySubject (same name, so Dynamic is added at the end) holds an object h of the ProxySubject class that implements the InvocationHandler interface. When the operation method of the proxy object is called, the invoke method of object h is called. The invoke method distinguishes the method based on the name of the method, and then calls the corresponding method of the specific object through the method.invoke() method.

Implementing ServiceHook using dynamic proxy in Android

Through the above introduction to InvocationHandler, we should have a general understanding of this interface, but what is the role of the proxy class dynamically generated at runtime? In fact, its role is to insert some additional operations before or after calling the real business:

So in short, the processing logic of the proxy class is very simple, which is to insert some additional services before and after calling a method. And our practical example in Android is to selectively do some special processing before and after actually calling a system service. This idea is also very important in the plug-in framework. So how do we specifically implement the hook system service and attach the services we need when actually calling the system service? This requires the introduction of the ServiceManager class.

Introduction to ServiceManager and hook steps

first step

The detailed introduction of ServiceManager has been introduced in my blog: android IPC communication (Part 2) - AIDL, so I will not repeat it here. I strongly recommend you to read that blog. Here we will focus on the getService(String name) method of ServiceManager:

  1. public final class ServiceManager {
  2. private static final String TAG = "ServiceManager" ;
  3.  
  4. private static IServiceManager sServiceManager;
  5. private static HashMap<String, IBinder> sCache = new HashMap<String, IBinder>();
  6.  
  7. private static IServiceManager getIServiceManager() {
  8. if (sServiceManager != null ) {
  9. return sServiceManager;
  10. }
  11.  
  12. // Find the service manager
  13. sServiceManager = ServiceManagerNative.asInterface(BinderInternal.getContextObject());
  14. return sServiceManager;
  15. }
  16.  
  17. /**
  18. * Returns a reference to a service with the given name .
  19. *
  20. * @param name the name   of the service to get
  21. * @ return a reference to the service, or <code> null </code> if the service doesn't exist
  22. */
  23. public   static IBinder getService(String name ) {
  24. try {
  25. IBinder service = sCache.get( name );
  26. if (service != null ) {
  27. return service;
  28. } else {
  29. return getIServiceManager().getService( name );
  30. }
  31. } catch (RemoteException e) {
  32. Log.e(TAG, "error in getService" , e);
  33. }
  34. return   null ;
  35. }
  36.  
  37. public   static void addService(String name , IBinder service) {
  38. ...
  39. }
  40. ....
  41. }

We can see that the first step of the getService method is to get the IBinder object of the Service according to the name of the Service in the sCache map. If it is empty, the IBinder object of the Service will be obtained through ServiceManagerNative through cross-process communication. So we use the sCache map as the entry point, reflect the object, and then modify the object. Since the system's android.os.ServiceManager class is @hide, only reflection can be used. Based on this preliminary idea, write the code for the first step:

  1. Class c_ServiceManager = Class.forName( "android.os.ServiceManager" );
  2. if (c_ServiceManager == null ) {
  3. return ;
  4. }
  5.  
  6. if (sCacheService == null ) {
  7. try {
  8. Field sCache = c_ServiceManager.getDeclaredField( "sCache" );
  9. sCache.setAccessible( true );
  10. sCacheService = (Map<String, IBinder>) sCache.get( null );
  11. } catch (Exception e) {
  12. e.printStackTrace();
  13. }
  14. }
  15. sCacheService.remove(serviceName);
  16. sCacheService.put(serviceName, service);

Reflect the sCache variable, remove the system Service, and then put our own modified Service into it. In this way, when the getService(String name) method of ServiceManager is called, the modified Service is returned instead of the system's native Service.

Step 2

The first step is to know how to put the modified Service into the system's ServiceManager. The second step is to generate a hook Service. How to generate it? This requires the use of the InvocationHandler class we introduced above. We first obtain the native Service, then use the InvocationHandler to construct a hook Service, and finally put it into the sCache variable through the first step. The second step code:

  1. public class ServiceHook implements InvocationHandler {
  2. private static final String TAG = "ServiceHook" ;
  3.  
  4. private IBinder mBase;
  5. private Class<?> mStub;
  6. private Class<?> mInterface;
  7. private InvocationHandler mInvocationHandler;
  8.  
  9. public ServiceHook(IBinder mBase, String iInterfaceName, boolean isStub, InvocationHandler InvocationHandler) {
  10. this.mBase = mBase;
  11. this.mInvocationHandler = InvocationHandler;
  12.  
  13. try {
  14. this.mInterface = Class.forName(iInterfaceName);
  15. this.mStub = Class.forName(String.format( "%s%s" , iInterfaceName, isStub ? "$Stub" : "" ));
  16. } catch (ClassNotFoundException e) {
  17. e.printStackTrace();
  18. }
  19. }
  20.  
  21. @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  22. if ( "queryLocalInterface" .equals(method.getName())) {
  23. return Proxy.newProxyInstance(proxy.getClass().getClassLoader(), new Class[] { mInterface },
  24. new HookHandler(mBase, mStub, mInvocationHandler));
  25. }
  26.  
  27. Log.e(TAG, "ERROR!!!!! method:name = " + method.getName());
  28. return method.invoke(mBase, args);
  29. }
  30.  
  31. private static class HookHandler implements InvocationHandler {
  32. private Object mBase;
  33. private InvocationHandler mInvocationHandler;
  34.  
  35. public HookHandler(IBinder base, Class<?> stubClass,
  36. InvocationHandler InvocationHandler) {
  37. mInvocationHandler = InvocationHandler;
  38.  
  39. try {
  40. Method asInterface = stubClass.getDeclaredMethod( "asInterface" , IBinder.class);
  41. this.mBase = asInterface.invoke( null , base);
  42. } catch (Exception e) {
  43. e.printStackTrace();
  44. }
  45. }
  46.  
  47. @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  48. if (mInvocationHandler != null ) {
  49. return mInvocationHandler.invoke(mBase, method, args);
  50. }
  51. return method.invoke(mBase, args);
  52. }
  53. }
  54. }

Here we take the calling code of ClipboardService as an example:

  1. IBinder clipboardService = ServiceManager.getService(Context.CLIPBOARD_SERVICE);
  2. String IClipboard = "android.content.IClipboard" ;
  3.  
  4. if (clipboardService != null ) {
  5. IBinder hookClipboardService =
  6. (IBinder) Proxy.newProxyInstance(clipboardService.getClass().getClassLoader(),
  7. clipboardService.getClass().getInterfaces(),
  8. new ServiceHook(clipboardService, IClipboard, true , new ClipboardHookHandler()));
  9. //Call the first step method
  10. ServiceManager.setService(Context.CLIPBOARD_SERVICE, hookClipboardService);
  11. } else {
  12. Log.e(TAG, "ClipboardService hook failed!" );
  13. }

Let's analyze the above code. Before analyzing it, we need to look at the relevant class diagram of ClipboardService:

From this class diagram, we can clearly see the inheritance relationship of ClipboardService. The next step is to analyze the code:

  • In the calling code, the second parameter of the Proxy.newProxyInstance function needs attention. Since ClipboardService inherits from three interfaces, all interfaces need to be passed in here. However, if the second parameter is changed to new Class[] { IBinder.class }, there is actually no problem (those who are interested can try it. The first parameter is changed to IBinder.class.getClassLoader(), and the second parameter is changed to new Class[]{IBinder.class}, which is also OK). In actual use, we only use the queryLocalInterface method of the IBinder class, and other methods are not used. Next, we will explain the role of the queryLocalInterface function;
  • The third parameter of the Proxy.newProxyInstance function is the ServiceHook object, so when this Service is set into the sCache variable, all operations calling ClipboardService will be called into the invoke method in ServiceHook;
  • Next, let's take a look at the invoke function of ServiceHook. It is very simple. When the function is the queryLocalInterface method, it returns a HookHandler object. In other cases, it directly calls the method.invoke native system ClipboardService function. Why only process the queryLocalInterface method? I have mentioned this in my blog: Java/Android Design Pattern Learning Notes (9) - Proxy Mode when analyzing AMS. The asInterface method will eventually call the queryLocalInterface method. The final return result of the queryLocalInterface method will be returned to the caller of the Service as the result of asInterface. Therefore, the object returned by the queryLocalInterface method will be directly called externally. This also explains why there is no problem with changing the second parameter in the calling code to new Class[] { IBinder.class }, because after the first call to the queryLocalInterface function, all subsequent calls go to the HookHandler object. The dynamically generated object only needs the queryLocalInterface method of IBinder, and does not need IClipboard. Other methods of the interface;
  • Next is the HookHandler class. First, let's look at the constructor of this class. The first parameter is the system's ClipboardService, and the second parameter is
  1. Class.forName(String.format( "%s%s" , iInterfaceName, isStub ? "$Stub" : "" ))// "android.content.IClipboard"  

Let's compare this parameter with the class diagram above. This class is the parent class of ClipboardService. It has an asInterface method. By reflecting the asInterface method, the IBinder object is converted into an IInterface object. To learn why this is done, you can read my blog: Java/Android Design Pattern Learning Notes (9) - Final Summary of the Proxy Pattern. An IBinder object is obtained through the ServiceManager.getService method, but this IBinder object cannot be called directly. It must be converted into the corresponding IInterface object through the asInterface method before it can be used. Therefore, the mBase object is actually an IInterface object:

Finally, this result is confirmed. I don’t need to explain why it is a Proxy object;

  • Finally, there is the invoke method of HookHandler, which calls the ClipboardHookHandler object. Let's take a look at the implementation of this class:
  1. public class ClipboardHook {
  2.  
  3. private static final String TAG = ClipboardHook.class.getSimpleName();
  4.  
  5. public   static void hookService(Context context) {
  6. IBinder clipboardService = ServiceManager.getService(Context.CLIPBOARD_SERVICE);
  7. String IClipboard = "android.content.IClipboard" ;
  8.  
  9. if (clipboardService != null ) {
  10. IBinder hookClipboardService =
  11. (IBinder) Proxy.newProxyInstance(IBinder.class.getClassLoader(),
  12. new Class[]{IBinder.class},
  13. new ServiceHook(clipboardService, IClipboard, true , new ClipboardHookHandler()));
  14. ServiceManager.setService(Context.CLIPBOARD_SERVICE, hookClipboardService);
  15. } else {
  16. Log.e(TAG, "ClipboardService hook failed!" );
  17. }
  18. }
  19.  
  20. public   static class ClipboardHookHandler implements InvocationHandler {
  21.  
  22. @Override
  23. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  24. String methodName = method.getName();
  25. int argsLength = args.length;
  26. //Every time you copy text from this app, add the source of the sharing
  27. if ( "setPrimaryClip" .equals(methodName)) {
  28. if (argsLength >= 2 && args[0] instanceof ClipData) {
  29. ClipData data = (ClipData) args[0];
  30. String text = data.getItemAt(0).getText().toString();
  31. text += "this is shared from ServiceHook-----by Shawn_Dut" ;
  32. args[0] = ClipData.newPlainText(data.getDescription().getLabel(), text);
  33. }
  34. }
  35. return method.invoke(proxy, args);
  36. }
  37. }
  38. }

Therefore, the invoke method of the ClipboardHookHandler class finally obtains the IInterface object of the Service to be hooked (that is, the IClipboard.Proxy object, and finally calls the system's ClipboardService through the Binder Driver), the Method object and the parameter list object of the calling function. After obtaining these, needless to say, you can do some additional operations as much as you want. Here I am imitating Zhihu's copying of text and adding a similar copyright statement at the end.

Discussion Questions

The above are the detailed steps of ServiceHook. To understand it, you must have a detailed understanding of InvocationHandler and look at the AOSP source code. For example, if you want to hook ClipboardService, you must first look at the source code of ClipboardService to see the name and function of each function in this class, the order and function of each parameter in the parameter list, and sometimes this is far from enough. We know that with the update of each version of Android, these classes may also be updated, modified or even deleted. It is very likely that the old hook method will no longer work for the new version. At this time, you must understand the latest source code, see the updated and modified places, and redefine the hook steps for the new version. This is a point that needs to be treated with caution.

step

Source code

This is the code of a great god in our company. It has been sorted and modified. I don't know if there are any copyright issues. Hahaha, thank you Zhou Jie, although he is no longer in the company, thank you very much~~

Source code download address: https://github.com/zhaozepeng/ServiceHook

Let's take a look at the effect of running:

Please indicate the source for reprinting: http://blog.csdn.net/self_study/article/details/55050627

<<:  What are machine learning and deep learning? Faizan Shaikh will help you answer

>>:  Android imitates Huawei's weather drawing dial

Recommend

Internet Product Manager From Beginner to Master Course

This course starts with the basics and explains i...

Alibaba App Marketing Brand Cooperation Application Process!

1. Introduction to brand cooperation Different fr...

Velocity Official Guide - Using Velocity

[[143748]] If you are using VelocityViewServlet o...

Tencent Guangdiantong advertising optimization strategy!

In the 2020s, is there still anyone who doesn’t k...

Analysis of 4 factors of Keep user experience!

A week ago, QuestMobile released the "China ...

Brand ≠ IP, operators need to have a deep understanding of IP and brand

Brand is not equal to IP, there is a clear differ...

8 ways to drive traffic to public accounts through short videos

Friends who work on short video columns are no st...

A perfect event planning solution cannot be without these elements!

Do you hope that the event will become a hit? Tha...

What are the characteristics of a reliable SEO project?

As an individual webmaster, how to use SEO to cre...

Baijiahao operation guide, how to operate Baijiahao?

1. Which one is better in the Baijiahao field? (F...

50% of the creative copywriting for information flow ads is incomprehensible!

About 50% of the creative copywriting in informat...