AOP在移动开发中的应用
2020-01-21
AOP在移动开发中的应用
下面是我在我们团队上的技术分享内容,从PPT里摘抄出来的,所有很多都是个提纲摘要
AOP简介
- AOP,Aspect Oriented Programming,面向切面编程
- 面向切面编程是一种通过横切关注点(Cross-cutting Concerns)分离来增强代码模块性的方法,它能够在不修改业务主体代码的情况下,对它添加额外的行为。
- 是对OOP的一种补充,是一种解耦的重要手段
AOP常用概念
- Join point:程序执行期间的一个点,表示方法的执行
- Pointcut: 切入点实际上也是从所有的连接点(Join point)挑选自己感兴趣的连接点的过程
- Aspect: 程序横向切割成若干的面,即Aspect.每个面被称为切面
- Advice: 某个特定连接点的某个方面采取的行动。不同类型的建议包括“周围(Around)”,“之前(Before)”和“之后(After)”建议
AOP实现原理
-
编译期间的静态织入,又称为编译时增强
-
运行期间的动态代理,又称为运行时增强
运行时AOP
- 程序运行时,依靠预先创建或运行时创建的代理类来完成切面功能。这种方式依赖编程语言的动态能力
- JDK 基于接口的动态代理技术(Spring AOP),基于Java反射特性实现
- Objective-C的Method Swizzling和KVO,基于OC的Runtime实现
JDK 基于接口的动态代理技术
基于接口编程
import java.text.MessageFormat;
import java.util.Date;
interface BookService {
void lendOut(String bookId, String userId, Date date);
}
class BookServiceImpl implements BookService {
@Override
public void lendOut(String bookId, String userId, Date date) {
System.out.println(MessageFormat.format("{0}: The book {1} is lent to {2}.", date, bookId, userId));
}
}
基于反射调用
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
class ServiceInvocationHandler implements InvocationHandler {
private Object target;
public ServiceInvocationHandler(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Before...");
Object result = method.invoke(this.target, args);
System.out.println("After...");
return result;
}
}
不直接持有子类实例,持有接口实例,真正的实例对象有别人创建
import java.lang.reflect.Proxy;
import java.util.Date;
public class Client {
public static void main(String[] args) throws Exception {
BookService bookService = (BookService) Proxy.newProxyInstance(
BookService.class.getClassLoader(),
new Class[]{ BookService.class },
new ServiceInvocationHandler(new BookServiceImpl())
);
bookService.lendOut("123", "456", new Date());
}
}
Objective-C的Method Swizzling
NSThread的start方法扩展
- (void)validStart {
@try {
if (self.isCancelled || self.isFinished || self.isExecuting) {
NSLog(@"should not start");
} else {
[self validStart]; //[self start];
}
} @catch (NSException *exception) {
NSLog(@"exception");
} @finally {
NSLog(@"@finally");
}
方法替换来实现AOP(语言本身支持runtime,非常强大的特性)
+ (void)load {
[super load];
Class class = [self class];
Method oldMethod = class_getInstanceMethod(class, NSSelectorFromString(@"start"));
Method newMethod = class_getInstanceMethod(class, NSSelectorFromString(@"validStart"));
method_exchangeImplementations(oldMethod, newMethod);
}
编译期AOP
- 编译期就把切面代码和业务代码链接起来,需要编译器或编译工具支持
- 有些编程语言编译期生成中间代码的话,较容易支持这种AOP实现
- Java的AspectJ,Flutter的Dart,Android的Gradle插件Transform API
移动端语言对于AOP的支持
- Java:支持运行时AOP和编译时AOP
- Objective-C支持运行时AOP
- kotlin支持编译期AOP(它编译成字节码)
- swift支持运行时AOP(基于Objective-C的特性,局限于OC代码)
- Flutter Dart支持编译期AOP,编译中间产物是dill(修改编译工具)
AOP在移动端应用:无埋点统计
- 传统埋点代码涉及各个模块,有点繁琐又没有太大技术含量
- 无埋点采用AOP实现,解耦代码,且不容易漏统计。业务开发人员只需关注自己的业务代码
Android Gradle 工具在 1.5.0 版本后提供了 Transfrom API, 允许第三方 Plugin 在打包 dex 文件之前的编译过程中操作 .class 文件 遍历所有要编译的class文件并对其中符合条件的方法进行修改,注入我们要调用的SDK数据搜集代码,从而实现自动埋点的目的