See the strong insertion of AspectJ in Android

See the strong insertion of AspectJ in Android

What is AOP

AOP is the abbreviation of Aspect Oriented Programming. It is a different idea from the OOP we usually come into contact with. OOP, or "object-oriented programming", advocates modularization and objectification of functions, while the idea of ​​AOP is different. It advocates unified processing of the same type of problems. Of course, in the actual programming process, we cannot simply use the ideas of AOP or OOP to program. In many cases, multiple programming ideas may be mixed. There is no need to worry about which idea to use. Taking the best of all schools is the right way.

So what is the use of AOP as a programming concept? Generally speaking, it is mainly used in scenarios where you do not want to invade the original code. For example, the SDK needs to insert some code into the host without intrusion to do logging, performance monitoring, dynamic permission control, and even code debugging.

AspectJ

AspectJ is actually a practice of AOP programming ideas. Of course, in addition to AspectJ, there are many other AOP implementations, such as ASMDex, but the most popular and convenient one is still AspectJ.

Using AspectJ in Android Projects

AOP is widely used, from Spring to Android, and is used everywhere, especially in the backend. It is very convenient to use in Spring and its functions are very powerful. However, in Android, the implementation of AspectJ is a slightly castrated version, and not all functions are supported. But for general client development, it is completely sufficient.

Integrating AspectJ on Android is actually quite complicated. It is not possible to compile it in one sentence. However, our company has solved this problem for you. Now you can use AspectJ in Android Studio very conveniently by using this SDK. The Github address is as follows:

https://github.com/HujiangTec…

Another successful library that uses AOP is Hugo by Jake:

https://github.com/JakeWharto…

Access Instructions

First, you need to add dependencies to the build.gradle file in the project root directory:

  1. classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:1.0.8'  

The complete code is as follows:

  1. buildscript {
  2. repositories {
  3. jcenter()
  4. }
  5. dependencies {
  6. classpath 'com.android.tools.build:gradle:2.3.0-beta2'  
  7. classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:1.0.8'  
  8.  
  9. // NOTE: Do not place your application dependencies here; they belong
  10. // in the individual module build.gradle files
  11. }
  12. }

Then add AspectJ dependency in the build.gradle of the main project or library:

  1. compile 'org.aspectj:aspectjrt:1.8.9'  

At the same time, add the AspectJX module to build.gradle:

  1. apply plugin: 'android-aspectjx'  

This completes the configuration of the AspectJ environment in the entire Android Studio. If you encounter errors such as "can't determine superclass of missing type xxxxx" during compilation, please refer to the project README for information about the use of excludeJarFilter.

  1. aspectjx {
  2. //includes the libs that you want to weave
  3. includeJarFilter 'universal-image-loader' , 'AspectJX-Demo/library'  
  4.  
  5. //excludes the libs that you don't want to weave
  6. excludeJarFilter 'universal-image-loader'  
  7. }

Getting started with AspectJ

Let's use a simple code to understand the basic usage and functions. Create a new AspectTest class file with the following code:

  1. @Aspect
  2. public class AspectTest {
  3.  
  4. private static final String TAG = "xuyisheng" ;
  5.  
  6. @Before( "execution(* android.app.Activity.on**(..))" )
  7. public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable {
  8. String key = joinPoint.getSignature().toString();
  9. Log.d(TAG, "onActivityMethodBefore: " + key );
  10. }
  11. }

At the beginning of the class, we use the @Aspect annotation to define such an AspectJ file. The compiler will automatically parse it during compilation, and there is no need to actively call the code in the AspectJ class.

My original code was simple:

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. }
  8. }

After compiling in this way, let's take a look at what the generated code looks like. The principle of AspectJ is actually to parse according to certain rules during compilation, and then insert some code. The code generated by aspectjx will be in the Build directory:

Use the decompilation tool to view the generated content:

We can find that a line of AspectJ code is inserted at the beginning of onCreate. This is the main function of AspectJ. Putting aside the idea of ​​AOP, what we want to do is actually "add new code without invading the original code."

AspectJ Join Points

Join Points, or JPoints for short, is one of the core ideas of AspectJ. It is like a knife that cuts the entire execution process of a program into different parts. For example, constructor calls, method calls, method executions, exceptions, etc., are all Join Points. In fact, it is where you want to insert new code in the program, whether it is in the constructor, before a method call, or in a method. This place is the Join Point. Of course, not all places can be inserted for you. Only places that can be inserted are called Join Points.

AspectJ Pointcuts

The difference between Join Points and Pointcuts is actually difficult to explain, and I dare not say that my understanding is necessarily correct, but these are all conceptual content and do not affect our use.

Pointcuts, as I understand it, is actually to select the Join Points we need through certain conditions in Join Points. Therefore, Pointcuts, that is, Join Points with conditions, serve as the code entry points we need.

AspectJ Advice

Here comes another Advice. Advice is actually the most basic understanding, that is, the specific code we insert and how to insert it. The example we gave at the beginning used the simplest Advice - Before. Similar ones include After and Around. We will talk about their differences later.

AspectJ pointcut syntax

Let's take a look at the simplest AspectJ syntax with the previous Demo:

  1. @Before( "execution(* android.app.Activity.on**(..))" )
  2. public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable {
  3. }

This will be divided into several parts, let's look at them one by one:

  • @Before: Advice, which is the specific insertion point
  • execution: Processes the type of Join Point, such as call and execution
  • ( android.app.Activity.on*(..)): This is the most important expression. The first '*' indicates the return value. '*' indicates that the return value is of any type. The following is a typical package name path, which can contain '*' for wildcarding. There is no difference between several '*'. At the same time, you can use '&&, ||, !' to combine conditions. () represents the parameters of this method. You can specify the type, such as android.os.Bundle, or (..) to represent any type and any number of parameters.
  • public void onActivityMethodBefore: the actual entry code.

Here are some matching rules that can be used as examples to explain:

expression meaning
java.lang.String Matches String type
java.*.String Matches the String type in any "first-level subpackage" under the java package, such as matching java.lang.String, but not matching java.lang.ss.String
java..* Matches any type in the java package and any subpackage, such as java.lang.String, java.lang.annotation.Annotation
java.lang.*ing Matches any type ending with ing in the java.lang package
java.lang.Number+ Matches any Number subtype under the java.lang package, such as java.lang.Integer and java.math.BigInteger
parameter meaning
() Indicates that the method does not have any parameters
(..) Indicates that the method that matches accepts any number of parameters
(..,java.lang.String) Indicates that the matching of the method that accepts a parameter of type java.lang.String ends, and the method that accepts any number of parameters can be matched.
(java.lang.String,..) Indicates that the method that matches the method that accepts a parameter of type java.lang.String and can accept any number of parameters is matched.
(*,java.lang.String) Indicates that the matching of the method that accepts a parameter of type java.lang.String ends and the method that accepts a parameter of any type is preceded by

AspectJ Example

Before、After

These two Advices are probably the most commonly used, so let's take a look at the examples of these two Advices, starting with Before and After.

  1. @Before( "execution(* com.xys.aspectjxdemo.MainActivity.on*(android.os.Bundle))" )
  2. public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable {
  3. String key = joinPoint.getSignature().toString();
  4. Log.d(TAG, "onActivityMethodBefore: " + key );
  5. }
  6.  
  7. @After ( "execution(* com.xys.aspectjxdemo.MainActivity.on*(android.os.Bundle))" )
  8. public void onActivityMethodAfter(JoinPoint joinPoint) throws Throwable {
  9. String key = joinPoint.getSignature().toString();
  10. Log.d(TAG, "onActivityMethodAfter: " + key );
  11. }

After the above grammar explanation, it should be easy to understand now. Let's take a look at the compiled class:

We can see that based on the original code, the Before and After codes are added, and the Log can be correctly inserted and printed out.

Around

Before and After are actually quite easy to understand, that is, inserting code before and after Pointcuts. What about Around? From the literal meaning, it means inserting code before and after the method. Yes, it contains all the functions of Before and After. The code is as follows:

  1. @Around( "execution(* com.xys.aspectjxdemo.MainActivity.testAOP())" )
  2. public void onActivityMethodAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
  3. String key = proceedingJoinPoint.getSignature().toString();
  4. Log.d(TAG, "onActivityMethodAroundFirst: " + key );
  5. proceedingJoinPoint.proceed();
  6. Log.d(TAG, "onActivityMethodAroundSecond: " + key );
  7. }

Among them, proceedingJoinPoint.proceed() represents the execution of the original method. Various logical processing can be performed before and after this method.

Original code:

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. testAOP();
  8. }
  9.  
  10. public void testAOP() {
  11. Log.d( "xuyisheng" , "testAOP" );
  12. }
  13. }

Let's take a look at the compiled code first:

We can find that Around does implement the functions of Before and After, but it should be noted that Around and After cannot act on the same method at the same time, which will cause the problem of repeated entry.

Custom Pointcuts

Custom Pointcuts allow us to more accurately cut into one or more specified entry points.

First, we need to customize an annotation class, for example - DebugTool.java:

  1. /**
  2. * Custom AOP annotations
  3. * <p>
  4. * Created by xuyisheng on 17/1/12.
  5. */
  6.  
  7. @Retention(RetentionPolicy.CLASS)
  8. @Target({ElementType.CONSTRUCTOR, ElementType.METHOD})
  9. public @interface DebugTool {
  10. }

Then use this annotation where you need to insert the code:

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. testAOP();
  8. }
  9.  
  10. @DebugTool
  11. public void testAOP() {
  12. Log.d( "xuyisheng" , "testAOP" );
  13. }
  14. }

***, let's create our own entry file.

  1. @Pointcut( "execution(@com.xys.aspectjxdemo.DebugTool * *(..))" )
  2. public void DebugToolMethod() {
  3. }
  4.  
  5. @Before( "DebugToolMethod()" )
  6. public void onDebugToolMethodBefore(JoinPoint joinPoint) throws Throwable {
  7. String key = joinPoint.getSignature().toString();
  8. Log.d(TAG, "onDebugToolMethodBefore: " + key );
  9. }

First define Pointcut and declare the method name to be monitored, then add the entry code in Before or other Advice to complete the entry.

The compiled code is as follows:

In this way, we can easily monitor the specified Pointcut, thereby increasing the granularity of monitoring.

call and execution

In the AspectJ pointcut expression, we have used execution before. In fact, there is another type - call. So what is the difference between these two syntaxes? Let's try it out and we will know.

The cut code is still very simple:

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. testAOP();
  8. }
  9.  
  10. public void testAOP() {
  11. Log.d( "xuyisheng" , "testAOP" );
  12. }
  13. }

Let's look at execution first. The code is as follows:

  1. @Before( "execution(* com.xys.aspectjxdemo.MainActivity.testAOP(..))" )
  2. public void methodAOPTest(JoinPoint joinPoint) throws Throwable {
  3. String key = joinPoint.getSignature().toString();
  4. Log.d(TAG, "methodAOPTest: " + key );
  5. }

The compiled code is as follows:

Let's look at the call again. The code is as follows:

  1. @Before( "call(* com.xys.aspectjxdemo.MainActivity.testAOP(..))" )
  2. public void methodAOPTest(JoinPoint joinPoint) throws Throwable {
  3. String key = joinPoint.getSignature().toString();
  4. Log.d(TAG, "methodAOPTest: " + key );
  5. }

The compiled code is as follows:

In fact, it is clear at a glance when you compare them. Execution is in the method being cut into, and call is before or after calling the method being cut into.

For Call:

  1. Call (Before)
  2. Pointcut
  3. Pointcut Method
  4. }
  5. Call ( After )

For Execution:

  1. Pointcut
  2. execution(Before)
  3. Pointcut Method
  4. execution( After
  5. }

Pointcut filtering and withincode

In addition to the call and execution mentioned above, there is also a more commonly used withincode. This syntax is usually used to filter some entry point conditions for more precise entry control. We can refer to the following example:

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.activity_main);
  7. testAOP1();
  8. testAOP2();
  9. }
  10.  
  11. public void testAOP() {
  12. Log.d( "xuyisheng" , "testAOP" );
  13. }
  14.  
  15. public void testAOP1() {
  16. testAOP();
  17. }
  18.  
  19. public void testAOP2() {
  20. testAOP();
  21. }
  22. }

Both testAOP1() and testAOP2() call the testAOP() method. However, if we want to enter the code only when the testAOP2() method calls the testAOP() method, we need to use a combination of Pointcut and withincode to accurately locate the entry point.

  1. // In the testAOP2() method
  2. @Pointcut( "withincode(* com.xys.aspectjxdemo.MainActivity.testAOP2(..))" )
  3. public void invokeAOP2() {
  4. }
  5.  
  6. // When calling the testAOP() method
  7. @Pointcut( "call(* com.xys.aspectjxdemo.MainActivity.testAOP(..))" )
  8. public void invokeAOP() {
  9. }
  10.  
  11. // The previous conditions are met at the same time, that is, the testAOP() method is called in the testAOP2() method.
  12. @Pointcut( "invokeAOP() && invokeAOP2()" )
  13. public void invokeAOPOnlyInAOP2() {
  14. }
  15.  
  16. @Before( "invokeAOPOnlyInAOP2()" )
  17. public void beforeInvokeAOPOnlyInAOP2(JoinPoint joinPoint) {
  18. String key = joinPoint.getSignature().toString();
  19. Log.d(TAG, "onDebugToolMethodBefore: " + key );
  20. }

Let's look at the compiled code again:

We can see that the code is inserted only in the testAOP2() method, which achieves precise conditional insertion.

Exception handling AfterThrowing

AfterThrowing is a relatively rare advice, which is used to handle unhandled exceptions in the program. Remember, this is very important, it is an unhandled exception. The specific reason will be known after we look at the decompiled code. Let's write an exception casually, the code is as follows:

  1. public void testAOP() {
  2. View   view = null ;
  3. view .animate();
  4. }

Then use AfterThrowing to write AOP code:

  1. @AfterThrowing(pointcut = "execution(* com.xys.aspectjxdemo.*.*(..))" , throwing = "exception" )
  2. public void catchExceptionMethod(Exception exception) {
  3. String message = exception.toString();
  4. Log.d(TAG, "catchExceptionMethod: " + message);
  5. }

This code is very simple. It also uses a similar expression as before, but here we use *.* for wildcarding to handle exceptions. In the exception, we execute a line of log. The compiled code is as follows:

We can see that all methods under the com.xys.aspectjxdemo package have been added with try catch. At the same time, the code we cut into has been inserted in the catch. However, it will still throw e, that is, the exception has been thrown and the crash will still occur. At the same time, if your original code has already tried catch, it will also fail to handle. For the specific reason, let's look at a decompiled code:

As you can see, in fact, there is another layer of try catch in the catch of the original code, so when e.printStackTrace() is try caught, no more exceptions will occur and it will be impossible to cut in.

AspectJX Use Cases

At present, many projects of our company have used this AOP solution, such as AOP-based dynamic permission management, AOP-based business data tracking, AOP-based performance monitoring system, etc.

Now a part of the source code of dynamic permission management based on AOP has been open sourced, but because of the need to separate the business code, this function code will be further improved later. You can continue to pay attention to it. The github address is as follows:

https://github.com/firefly112…

Other AOP projects are being open sourced one after another, so you can continue to pay attention~

<<:  Using RenderScript to achieve Gaussian blur (frosted glass/frosted) effect

>>:  Aiti Tribe Stories (7): Meet 51CTO and aim high

Recommend

Product Operations: How to Essentially Explore Product Growth?

Don't panic when you encounter a cold winter....

Summary of Common Methods for Custom Controls

[[183466]] In addition to the measurement, layout...

Case Study: 12 Gamification Strategies for User Growth

This article uses a case study to introduce how t...

Photography class by Chen Yu

Introduction to Chen Yu’s photography class resou...

Hprose 2.0.0 for HTML5 released, high-performance cross-language RPC

[[145117]] Hprose 2.0.0 for HTML5 is finally rele...

Two marketing warnings during the epidemic

After the outbreak of the new coronavirus pneumon...

Finally, we've got you! WeChat for Android will be updated with dark mode

March 10th news, recently, the dispute between Te...

4 key steps to good event planning

When it comes to event planning , perhaps in many...

Xiaohongshu KOL content operation strategy from 1 to 100

Users' following, comments, unfollowing, etc....