public interface Animal{
void eat();
}
public class Cat implements Animal{
@Override
public void eat() {
Console.log("猫吃鱼");
}
}
TimeIntervalAspect这个切面代理上述对象,来统计猫吃鱼的执行时间:Animal cat = ProxyUtil.proxy(new Cat(), TimeIntervalAspect.class);
cat.eat();
TimeIntervalAspect位于cn.hutool.aop.aspects包,继承自SimpleAspect,代码如下:
public class TimeIntervalAspect extends SimpleAspect{
//TimeInterval为Hutool实现的一个计时器
private TimeInterval interval = new TimeInterval();
@Override
public boolean before(Object target, Method method, Object[] args) {
interval.start();
return true;
}
@Override
public boolean after(Object target, Method method, Object[] args) {
Console.log("Method [{}.{}] execute spend [{}]ms", target.getClass().getName(), method.getName(), interval.intervalMs());
return true;
}
}
执行结果为:
猫吃鱼
Method [cn.hutool.aop.test.AopTest$Cat.eat] execute spend [16]ms
在调用proxy方法后,IDE自动补全返回对象为Cat,因为JDK机制的原因,我们的返回值必须是被代理类实现的接口,因此需要手动将返回值改为Animal,否则会报类型转换失败。
使用Cglib的好处是无需定义接口即可对对象直接实现切面,使用方式完全一致:
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.2.7</version>
</dependency>
public class Dog {
public String eat() {
Console.log("狗吃肉");
}
}
Dog dog = ProxyUtil.proxy(new Dog(), TimeIntervalAspect.class);
String result = dog.eat();
执行结果为:
狗吃肉
Method [cn.hutool.aop.test.AopTest$Dog.eat] execute spend [13]ms
ProxyUtil中还提供了一些便捷的Proxy方法封装,例如newProxyInstance封装了Proxy.newProxyInstance方法,提供泛型返回值,并提供更多参数类型支持。
动态代理对象的创建原理是假设创建的代理对象名为 $Proxy0: